Java技术体系的自动内存管理,最根本的目标是自动化解决两个问题:自动给对象分配内存以及自动回收给对象的内存。
基本的内存分配原则:
- 对象优先在Eden分配
- 大对象直接进入老年代
- 长期存活的对象将进入老年代
- 空间分配担保
对象优先在Eden分配
大多数情况下,对象在新生代Eden区中分配。当Eden区没有足够空间进行分配时,虚拟机将发起一次Minor GC。
大对象直接进入老年代
大对象指需要大量连续内存空间的Java对象,最典型的大对象便是那种很长的字符串,或者元素数量很庞大的数组。
在JVM中要避免大对象的原因是,在分配空间时,它容易导致内存明明还有不少空间时就提前触发垃圾收集,以获得足够的连续空间才能安置好它们,而当复制对象时,大对象就意味着高额的内存复制开销。
HotSpot虚拟机提供了-XX:PretenureSizeThreshold参数,指定大于该设置值的对象直接在老年代分配,避免在Eden区及两个Survivor区之间来回复制,产生大量的内存复制操作。
(-XX:PretenureSizeThreshold该参数只对Serial和ParNew两款新生代收集器有效。)
长期存活的对象将进入老年代
JVM给每个对象定义了一个对象年龄(Age)计数器,存储在对象头中。
HotSpot虚拟机并不是永远要求对象的年龄必须达到-XX:MaxTenuringThreshold才能晋升到老年代,如果Survivor空间中相同年龄所有对象的大小的总和大于Survivor空间的一半,年龄大于或等于该年龄的对象就可以直接进入老年代。
空间分配担保
新生代使用复制-收集算法,但为了内存利用率,只使用其中一个Survivor空间来作为轮换备份,因此当出现大量对象在Minor GC后仍然存活的情况——最极端的情况就是内存回收后新生代中所有对象都存活,需要老年代进行分配担保,把Survivor无法容纳的对象直接送入老年代。
老年代要进行这样的担保,前提是老年代本身还有容纳这些对象的剩余空间(取之前每一次回收晋升到老年代对象容量的平均大小作为经验值,与老年代的剩余空间进行比较,决定是否进行Full GC来让老年代腾出更多的内存空间。)
《深入理解Java虚拟机》 第三版