本文主要是介绍JVM10_堆空间之新生代老年代、内存分配过程、Minor GC(Young GC)、Major GC、Full GC、TLAB、逃逸分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
要想设置新生代中的比例,必须显式指定SurvivorRatio, 只是关闭自适应分配策略不管用
对象内存分配过程
图解
自我总结过程:
首先new的对象会放在Eden区,当Eden区满了以后会触发YoungGC(Minor GC),此时会把没有被GC的对象放到S0或者S1区,假设此时就是放到了S0区。当对象被放到S区开始,就会为这些对象分配一个age年龄计数器(当到了一定年龄对象就会被放到老年区)
当Eden区再一次满了,又触发YGC,幸存的对象会被放到S1区,而之前S0区没有被YGC的对象也会放到S1区。
如此重复,当S区中对象到了一定年龄,就会被放到老年区
当Eden区满了会触发YGC,但是S0或者S1区满了不会触发YGC,Eden区满了以后执行YGC,此时S0或S1会被动一起跟着执行YGC
对象内存分配的特殊情况
新产生的对象,如果Eden区能放下就放,放不下就先进行YGC,对新生代进行垃圾回收(新生代中没有被回收的被放到to区),看回收后是否能放下,如果还是放不下就直接放到老年代。
老年代一般比新生代空间大,老年代如果放不下,那么就触发FullGC(或者Major GC)对老年代进行垃圾回收。
如果还是放不下,那么就报OOM
当然JVM默认是有动态扩容机制的,当老年代内存不够会扩容,也可以设置为固定大小,此时放不下就OOM
在进行YGC时,S区也被动进行GC,然后对象就会被放到to区,如果to区放不下,那么就会直接晋升老年代,而不会等到age到了阈值
Minor GC(Young GC)、Major GC、Full GC
主要注意区分major GC 和 FullGC
有时候直接就叫majorGC。要问清楚是问老年代的major GC,还是整堆的full GC
Minor GC触发的STW(Stop The World)时间很短,而另外两个GC引发的STW时间很长,所以JVM优化大部分是针对MajorGC和FullGC的
堆空间分代的意义
内存分配策略
new出来的东西肯定是放到堆空间的
TLAB
所以说,堆空间并不是都是共享的,每个线程都可以有一块独立的空间TLAB
JVM参数总结
堆并不是对象分配的唯一位置
如果一个对象只在方法中使用,没有逃逸出方法,那么就可能被分配到栈上
逃逸分析
判断是否发生逃逸,就看new出来的对象是否在方法外被调用
基于逃逸分析,所以我们在定义变量的时候,能使用局部变量,就不要在方法外定义
这样就会优先进行栈上分配内存,栈上分配的内存方法结束后也就直接出栈了,不会存在GC,
逃逸分析后栈上分配的优势 : 1、不存在GC 2、运行速度更快
逃逸分析,代码优化
标量替换
把一个聚合量替换成标量的表示形式,既然成为标量了,那么就可以放在栈帧中的局部变量表中,不需要在堆上分配,减少了GC
逃逸分析并不是很成熟,
TaobaoVM中的GCIH技术(将经常用的对象直接放到了不进行垃圾回收的区域)已经很成熟,只是增加了硬件,所以可以实现
HotSpot JVM中,虽然有逃逸分析,但是并没有使用栈上分配。而只是使用了标量替换,把聚合量拆解成多个标量,而标量就可以在栈上
所以最终明确,我们使用的hotspot虚拟机,对象实例都是在堆中分配
这篇关于JVM10_堆空间之新生代老年代、内存分配过程、Minor GC(Young GC)、Major GC、Full GC、TLAB、逃逸分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!