JVM中对象内存的分配策略

版权声明:转载请注明原文链接 https://blog.csdn.net/NVPS_wyj/article/details/82315128

Java中的自动内存管理主要解决了两个问题:内存的分配和内存的回收。
对象的分配回收策略:

  1. 优先分配Eden

    现在主流的垃圾回收算法是分代回收算法,内存被分为新生代和老年代,新生代中为了防止内存碎片化,一般采用复制算法。创建的对象优先在新生代的Eden区进行分配,当Eden区的内存不够存储需要分配的对象时,虚拟机将会发生一次新生代的minorGC。如果回收之后仍然没有足够的内存进行对象分配,则会触发分配担保机制,将新生代中的对象转移到老年代中。

  2. 大对象直接进入老年代
    大对象指的是需要大量连续内存的对象,比如很长的字符串以及数组。JVM提供了-XX:PretenureSizeThreshold参数来设置大对象的内存的值,当对象内存大于等于这个值时,该对象会直接被分配到老年代。这样做的目的是防止新生代发生大量的内存拷贝,因为新生代采用的是复制算法。

  3. 长期存活对象进入老年代
    JVM使用分代回收算法来管理内存,则需要明确哪些对象应该在新生代,哪些对象应该在老年代。因此JVM给每一个对象定义了一个对象年龄计数器。对象最初被分配到Eden区,进行第一次minorGC仍然存活后,如果能被survivor区接纳,则将对象移动到survivor区中,并将对象的年龄+1。当对象达到一定年龄后,就会被转移到老年代中。可以通过参数-XX:MaxTenuringThreshold参数设置年龄的阈值,默认为15。

  4. 动态对象年龄判定
    为了更好的适应不同程序的内存情况,JVM并不要求对象的年龄达到阈值才被移动到老年代中。如果在survivor区中相同年龄的对象所占用的内存超过survivor内存的一半,JVM会将大于等于该年龄的对象转移到老年代中。

  5. 分配担保机制
    在发生minorGC时,JVM会检测之前每次转移到老年代的大小的平均值,如果该平均值大于老年代的剩余空间,则发生fullGC;如果小于,则查看HandlePromotionFailure是否允许担保失败,如果允许,则只发生minorGC,否则发生fullGC。

猜你喜欢

转载自blog.csdn.net/NVPS_wyj/article/details/82315128