浅谈Java堆内存分代回收

概述

与C++不同的是:在Java中我们无需关心对象占用空间的释放,这主要得益于Java中的垃圾处理器(简称GC)帮助我们自动的进行对象占用空间的释放。

下面我们带着几个问题来学习:

  1. 堆内存是如何分代的?
  2. 各分代之间是如何配合工作的?

1、堆内存是如何分代的?

用一张图片来描述(面积大小不代表实际占用空间大小)

堆内存分为:年轻代(Young) + 老年代(Old),年轻代又分为:Eden区 + Survivor区 * 2。

通常年轻代中的各区比值为:Eden区:Survivor0 :Survivor1 = 8:1:1

要尽可能的让对象不进入Old区。

S0和S1默认情况下会动态的自动调整大小,可以使用`-XX:UseAdaptiveSizePolicy`来关闭动态调整

2、各分代之间是如何配合工作的?

以一个对象的在每个区之间的复制来描述这个问题,我们假定这个对象一直存活着。

当我们写一段如下代码时

User user = new User();

user对象首先被放入到Eden区,当Eden区满的时候会发生第一次Minor GC,这时垃圾收集器会在S0和S1中随机选择一个区(假设选中了S0)来存放Eden区剩余存活的对象。当Eden区再次满的时候会发生第二次Minor GC,这时垃圾收集器会把Eden区存活的对象 + S0中存活的对象复制到S1中,当Eden区再次满的时候会发生第三次Minor GC,这时垃圾收集器会把Eden区存活的对象 + S1中存活的对象复制到S0中,如此循环复制。

那么对象什么情况下会进入到Old区?

  • Eden区满时,在对象将要进入S0或S1中时,如果S0或S1存放不下Eden区中某个对象时,这个对象将被复制到Old区。
  • 如果一个对象在S0和S1中经历了制定次数的复制之后,会被复制到Old区。

顺便说一下:年轻代垃圾回收使用复制算法,老年代回收使用标记清除算法。

 

posted on 2018-07-14 16:55 吴庆龙 阅读() 评论() 编辑 收藏

版权声明:本文为wuqinglong原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/wuqinglong/p/9310136.html