堆内存分配策略傻傻不清楚

程序员小迷 2024-05-04 12:00:56

一、概述

1.JVM堆内存分为年轻代、老年代和持久代(JDK7及之前版本)或元空间(JDK8及之后版本)。

年轻代用于存放新创建的对象,老年代用于存放存活时间较长的对象。

持久代或元空间主要用于存放类信息、方法信息、常量池等。持久代在JDK7及之前的版本中容易出现内存溢出,因此在JDK8之后被元空间所替代。

2.年轻代包括一个Eden区和2个Survior区。对象优先在Eden分配,若Eden内存空间不足,就会发生Minor GC,存活的对象会移到另一个Survior区。多次Minor GC后仍然存活的对象会晋升到老年代。

3.大对象直接进入老年代。

大对象:需要大量连续内存空间的Java对象,比如很长的字符串和大型数组:

1)需要内存有空间,还是需要提前进行垃圾回收以获取连续空间来存放大对象。

2)会进行大量的内存复制。

-XX:PretenureSizeThreshold 参数 ,大于这个数量直接在老年代分配,缺省为0 ,表示绝不会直接分配在老年代。

4.长期存活的对象将进入老年代。默认15岁,可由JVM参数-XX:MaxTenuringThreshold调整。

5.动态对象年龄判定。为了能更好地适应不同程序的内存状况,虚拟机并不是永远地要求对象的年龄必须达到了MaxTenuringThreshold才能晋升老年代。如果在Survivor空间中相同年龄所有对象大小的总和大于Survivor空间的一半,年龄大于或等于该年龄的对象就可以直接进入老年代,无须等到MaxTenuringThreshold中要求的年龄。

6.空间分配担保:新生代中有大量的对象存活,Survivor空间不够,当出现大量对象在Minor GC后仍然存活的情况(最极端的情况就是内存回收后新生代中所有对象都存活),就需要老年代进行分配担保,把Survivor无法容纳的对象直接进入老年代。只要老年代的连续空间大于新生代对象的总大小或者历次晋升的平均大小,就进行Minor GC,否则进行Full GC。

二、总结

1.JVM在创建对象时,会根据对象的类型和大小来决定在堆内存中的哪个区域进行分配。

2.对象优先分配在Eden区。

3.大对象直接进入老年代。

4.长期存活的对象将进入老年代。

5.动态对象年龄判定。

6.空间分配担保。

微风不燥,阳光正好,你就像风一样经过这里,愿你停留的片刻温暖舒心。

我是程序员小迷(致力于C、C++、Java、Kotlin、Android、Shell、JavaScript、TypeScript、Python等编程技术的技巧经验分享),若作品对您有帮助,请关注、分享、点赞、收藏、在看、喜欢,您的支持是我们为您提供帮助的最大动力。

欢迎关注。助您在编程路上越走越好!

0 阅读:15