JVM内存模型(二)
1、内存模型以及分区
JVM分为堆区和栈区,还有方法区,初始化的对象放在堆里面,引用放在栈里面,class类信息常量池(class常量和static常量)等放在方法区
- 方法区:主要是存储类信息,常量池(static常量和static变量);编译后的代码(字节码)等数据
- 堆:初始化的对象,成员变量(那种非static的变量);所有的对象实例和数组都要在堆上分配
- 栈:栈的结构是由栈帧组成的,调用一个方法就压入一帧,帧上面存储局部变量表,操作数栈,动态链接,方法返回地址等信息,局部变量表存放的是8大基础类型加上一个引用类型,所以还是一个指向地址的指针
- 本地方法栈:主要为 Native 方法服务
- 程序计数器:记录当前线程执行的行号
堆里面的分区:Eden,survival(from+to),老年代,各自的特点
- 堆里面分为新生代和老年代,java8取消了永久代,采用了MetaSpace(元空间)
- 新生代包含 Eden + Survivor 区,Survivor 区里面分为 from + to 区,内存回收时,如果用的是复制算法,从 from 复制到 to,当经过一次或者多次 GC 之后,存活下来的对象会被移动到老年代,或者大对象直接在老年代分配内存
- 当 JVM 内存不够用的时候,会触发 full GC,清理 JVM 老年代和元空间,当新生代满了之后会触发 Young GC,先把存活的对象放到其中一个 Survivor区,然后进行垃圾清理
- 大对象在老年代直接分配内存,需要老年代的担保,也就是有足够的空间进行大对象的内存分配
- 存活的对象每次从 from 到 to 区,该对象的 GC 年龄阈值 + 1,一般这个阈值为 15(MaxTenuringThreshold)