JVM内存模型:
- 线程共享
- 方法区:
方法区同堆一样,是所有线程共享的内存区域,为了区分堆,又被称为非堆。用于存储已被虚拟机加载的类信息、常量、静态变量,如static修饰的变量加载类的时候就被加载到方法区中。运行时常量池是方法区的一部分,class文件除了有类的字段、接口、方法等描述信息之外,还有常量池用于存放编译期间生成的各种字面量和符号引用。 -
堆:
堆是java虚拟机管理内存最大的一块内存区域,因为堆存放的对象是线程共享的,所以多线程的时候也需要同步机制。它是所有线程共享的,它的目的是存放对象实例。
1)年轻代收集器
Serial、ParNew、Parallel Scavenge
2)老年代收集器
Serial Old、Parallel Old、CMS收集器
CMS:收集器- 初始标记:标记一下GC Roots能直接关联到的对象,速度很快
- 并发标记:GC Roots Tarcing过程,即可达性分析
- 重新标记:为了修正因并发标记期间用户程序运作而产生变动的那一部分对象的标记记录,会有些许停顿,时间上一般 初始标记 < 重新标记 < 并发标记
- 并发清除
3)特殊收集器
G1收集器[新型,不在年轻、老年代范畴内]
- 方法区:
- 线程私有
- 程序计数器:
为了线程切换可以恢复到正确执行位置,每个线程都需有独 立的一个程序计数器,不同线程之间的程序计数器互不影响,独立存储。 - 本地方法栈:
本地方法栈是与虚拟机栈发挥的作用十分相似,区别是虚拟机栈执行的是Java方法(也就是字节码)服务,而本地方法栈则为虚拟机使用到的native方法服务,可能底层调用的c或者c++。 -
虚拟机栈
每个方法的执行和结束对应着栈帧的入栈和出栈。入栈表示被调用,出栈表示执行完毕或者返回异常。一个虚拟机栈对应一个线程,当前CPU调度的那个线程叫做活动线程;一个栈帧对应一个方法,活动线程的虚拟机栈里最顶部的栈帧代表了当前正在执行的方法,而这个栈帧也被叫做‘当前栈帧’。
- 局部变量表:
是一片逻辑连续的内存空间,最小单位是Slot,用来存放方法参数和方法内部定义的局部变量。 - 操作数栈:
1).栈桢刚创建时,里面的操作数栈是空的。
2).Java虚拟机提供指令来让操作数栈对一些数据进行入栈操作,比如可以把局部变量表里的数据、实例的字段等数据入栈。
3).同时也有指令来支持出栈操作。
4).向其他方法传参的参数,也存在操作数栈中。
5).其他方法返回的结果,返回时存在操作数栈中。 - 动态链接:
一个方法调用另一个方法,或者一个类使用另一个类的成员变量时,
符号引用和直接引用在运行时进行解析和链接的过程,叫动态链接。
每一个栈帧内部都要包含一个指向运行时常量池的引用,来支持动态链接的实现。 - 方法返回地址:
返回一个值给调用它的方法。当前栈桢就被用来恢复调用者的状态,都恢复哪些呢?恢复局部变量表、操作数栈 和 程序计数器(pc指针),而这个程序计数器要适当地增加,来指向下一条指令(也就是调用函数的下一句)。使调用者方法能够正常地继续执行下去,而且返回值push到了调用方法的操作数栈中。
- 局部变量表:
- 程序计数器:
最后总结一下:
参考:
深入理解JVM-内存模型(jmm)和GC
JVM内存堆布局图解分析