内存模型
JVM堆,栈,方法区,计数器
- 堆
线程共享
存放所有对象实例,GC的主要区域,可以分为新生代,老年代,新生代可以分为一个eden,两个survivor空间(from survivor,to survivor),不需要连续内存,使用-XMX和-XMS控制空间最大最小,推荐写一样大,减少堆空间切换过程的消耗
- 栈(本地方法栈,虚拟机栈)
线程私有
存放局部变量,操作数栈,动态链接,方法出口等信息
- 方法区(运行时常量池)
线程共享
存放类的基本信息,静态变量,常量,即时编译器编译后的代码等
- 计数器
线程私有
记录正在执行的虚拟机字节码指令的地址,native方法为空
- GC
线程共享
新生代使用复制算法
老年代使用标记-清理或标记-整理算法
- 内存分配:
- 大多数情况,对象直接在新生代的Eden空间分配,当eden空间不够时,发起Minor GC,因为新生代对象存活时间很短所以会频繁发生Minor GC,其执行速度也会比较快。
- 大对象会直接分配到老年代中,大对象指需要连续内存空间的对象,如很长的字符串或数组,可以通过指令
-XX:PretenureSizeThreshold
参数控制大小分界线,老年代会执行Full GC - 经过Minor GC依然存活,且能被Survivor区容纳的对象,会被转移到Survivor区,标记计数器就增加1,默认增加到15时会转移到老年代,可以通过
-XX:MaxTenuringThreshold
设置。 - JVM在执行Minor GC之前,会检查老年代剩余空间>新生代总空间,如果大于那么Minor GC是安全的,如果小于则会查看指令
HandlePromotionFailure
是否允许担保失败,如果允许则继续检查老年代剩余空间>历次晋升到老年代对象的平均大小,如果大于,则尝试进行Minor GC,如果小于或不允许,则进行一次FULL GC,使老年代清除一次垃圾 - 调用System.gc()也会触发Full GC,Full GC后空间仍然不足,则会抛出OutOfMemoryError,调优时,尽量让对象在Minor GC期间被回收,让对象在新生代多存活,不要使用大对象
网友评论