JVM内存模型
JVM组成
- 类装载子系统
- 字节码执行引擎
- 运行时数据区(内存模型)
JVM内存模型
- 本地方法栈(线程独立)
- 栈(线程独立)
设置一个线程栈大小参数:-Xss,默认1M
每个线程会在栈内存区分配一块内存给本线程使用,我们叫线程栈,每个方法是一个栈帧。栈帧里面大概分4部分:局部变量表、操作数栈、动态链接、方法出口。 - 程序计数器(线程独立)
- 堆(线程共享)
设置堆大小参数:-Xms -Xmx - 方法区也叫元空间(线程共享)
设置方法区大小参数:-XX:MetaspaceSize -XX:MaxMetaspaceSize
使用直接内存,上限可以无限大直至撑满物理内存。
默认值为21M。建议要设置此参数。因为初始值太小,方法区满会触发fullGC。然后回触发扩容。
GC
- GC Root:方法区静态变量、线程栈局部变量表、本地方法栈变量
- 堆中的区域分配:老年代(2/3)、新生代(Eden区(8),S0区(1),S1区(1))
new一个对象时,先判断是不是栈上分配(逃逸分析),如果不是判断是不是大对象(可以通过参数设置阈值),大对象直接进入老年代;不是大对象进入Eden区,如果Eden区放不下,触发一次monirGC,剩余存活对象进入S0,下次再触发minorGC时同时回收Eden和S0,剩余存货对象进入S1,以此交替。每存活一次年龄+1,达到最大年龄15(可以设置进入老年大的年龄),进入老年代。如果GC之后对象还是放不下,会触发FullGC,同时回收老年代、新生代和元数据区。对象进入Eden区时,也可能根据当前存活对象大小等判断,S区放不下时,也可能直接触发FullGC。 - 方法区回收的无用的类需满足的条件:
2.1 该类所有的实例对象被回收
2.2 加载该类的classloader已经被回收
2.3 该类的Class对象没有在任何地方引用,无法在任何地方通过反射访问该类
对象结构
- 对象头
1.1 Mark Word(32位4字节,64位8字节)
1.2 Klass Pointer类型指针(64位开启压缩4字节,关闭压缩8字节)类的元数据的指针
1.3 数组长度(4字节,只有数组对象才有) - 实例数据
- 对齐填充(保证是8字节的整数倍)