JVM内存模型解析----------运行时数据区域
今天为大家讲解一下JVM内存模型(运行时数据区域),主要从运行时数据区域的几个部分进行深入了解,各部分的功能以及联系。
一.运行时数据区域的各组成部分
程序计数器:这是一块很小的内存区域,可以把它当作当前线程所执行的字节码的行号指示器;每执行一条执行,程序计数器就会使其行号进行替换开始执行下一条指令;从图上可以程序计数器所显示的字节码指令是通过执行引擎进行执行的。
栈:它的生命周期与线程相同,如何所示
其中每个方法栈中包括局部变量表、操作数栈、动态链接、方法出口。
public class Math {
public static final Integer CONSTANT = 0;
public int compute(){
int a=1;
int b=2;
int c=(a+b)*5;
return c;
}
public static void main(String[] args) {
Math math = new Math();
math.compute();
}
}
通过对上述代码形成的class进行反编译javap -c Math.class可以清晰的了解到这四个的功能,如下图
iconst_1:将int类型常量1压入栈,也就是先把常量1放入到操作数栈中
istore_1:将int类型值存入局部变量,也就是从操作数栈中提取数值1放入到局部变量表中
iconst_2:将int类型常量2压入栈,也就是先把常量1放入到操作数栈中
istore_2:将int类型值存入局部变量,也就是从操作数栈中提取数值2放入到局部变量表中
iload_1:从局部变量中装载int类型值,也就是把1赋值给变量a
iload_2:从局部变量中装载int类型值,也就是把2赋值给变量b
iadd:执行int类型的加法,把变量值进行相加
iconst_5:将int类型常量5压入栈
imul:执行int类型的乘法
istore_3:将int类型值存入局部变量,也就是从操作数栈中提取数值3放入到局部变量表中
iload_3:从局部变量中装载int类型值,也就是把15赋值给变量c
而栈帧中所提到的方法出口如下图所示
从第一个箭头位置移动到第二个箭头位置,则运用到了方法出口,使其执行其他的方法。所谓动态链接则是程序在运行过程中各种各样的方法所生成的指令码,也就是程序在运行过程中所生成的地址。
本地方法栈:与普通栈相似,但普通java栈执行的是java方法,而本地方法栈则执行的是Native方法。
方法区:里面存储的静态变量、常量、类元信息(class文件的一些组成结构信息)。
堆:这是我们最重要的一块,也是最复杂的一块,如下图所示,堆其实是由三块区域构成
总体分为2个部分:新生代和老年代;而新生代中又分为Eden区和Survivor区,比例为8:1。对象主要存在于Eden区中,而Survivor区的作用就是减少被送到老年代的次数从而减少full gc发生的次数,因为相比较minor gc显然full gc停顿的时间更长,每进行一次minor gc年龄就+1,当达到15(如果用Parallel Scavenge+Parallel Old默认为15,CMS默认为6)的时候,此时的对象便会被送到老年代中,当老年代中的对象满了之后便会发生full gc或major gc的现象。