如果我错了,请随时指正。在JVM堆中,有老一代和年轻一代两代。在做全GC时,在老一代中,有像紧凑空间和修复漏洞这样的繁重操作,这会使JVM挂起。而我发现在年轻一代中,应用了一个轻量级的GC,从我的搜索结果中还有一个叫做Eden的区域涉及年轻一代。但是,在搜索了很多文档后,我对年轻一代中的GC仍然有两个困惑,
这是你必须记住和理解的最重要的图表:
它来自JavaSE6HotSpot[tm]虚拟机垃圾收集调优,一站式学习GC内部的一切。但为了解决你眼前的问题:
使用new
操作符分配新对象(几乎)总是发生在Eden空间中。但伊甸园实际上是一堆。当您创建需要N字节的新对象时,单指针在堆栈上前进N字节,仅此而已。分配是如此之快,没有寻找空闲点,压缩,等等。
当然,这个堆栈不是无限的,在某个时候我们会到达它的终点,触发小GC。而且很可能多个对象已经是垃圾了。因此,JVM在小型GC中的作用如下:
>
从GC根开始的对象遍历图
将所有可从GC根访问的对象复制到一个幸存者空间(没有间隙,我们知道所有这些,这是一个单一的过程)
清除伊甸园空间(基本上只是将这个堆栈指针移回0
)
在随后的次要集合中,还有其他步骤:
那么,在终身制的一代中,对象是如何结束的呢?第一个年轻的对象被复制到一个幸存者空间。然后它们被一次又一次地复制到另一个。一旦给定对象来回跳转太多次(可配置,默认为8次),它将被提升到终身空间。
当终身空间满时,主GC运行。
我在尝试理解垃圾收集机制,我在研究代际算法,我有一个关于年轻人和老年人的代沟的问题。我读到,在年轻一代开始收集物品,GC是从GC根开始标记它们,以找到活的,通常它会将它们复制到幸存者空间,清除年轻一代区域,然后瞧。 我不明白,如果我们从GC根开始,我们开始遍历活动对象,我们不是也在旧一代中找到了对象吗?这是否意味着,当我们击中旧空间中的一个物体时,我们会在那个点上停止跟踪参照物?
问题内容: 当伊甸园空间年轻时已满,将触发次要GC。在次要GC过程中,伊甸园和一个源Survivor空间中的非自由对象将被复制到另一个目标Survivor空间。 我的问题是,如果目标“幸存者”空间已满,次要GC如何处理? 问题答案: 如果不可能执行/完成次要收集,则将执行主要/完整收集。通常使用标记扫描紧凑算法而不是复制算法来完成此操作……这是完整收集昂贵的原因之一。 但是最终(如果您继续填充堆)
在收集年轻一代内存时,JVM收集器只扫描那些属于年轻一代的根对象(堆中的OBEJCT可从根集中直接访问),还使用写屏障支持的卡表/记忆集来确定旧一代的区域,这些区域可能包含对年轻一代中对象的引用的对象。 我的问题是,如果年轻的收集器确定年轻一代中的某个特定对象只有来自老一代中某个对象的一个外部引用,它如何知道老一代对象本身是否是垃圾,从而使年轻一代对象“活动”且不符合收集条件?例如,在老一代中,可
当伊甸园空间充满年轻一代时,小GC将被触发。在次要GC过程中,伊甸园和一个源幸存者空间中的非自由对象将被复制到另一个目标幸存者空间。 我的问题是,如果目标幸存者空间已满,minor GC如何处理?
主要内容:垃圾回收算法,1、垃圾回收器的分类,2、串行垃圾回收器,3、吞吐量优先,4、响应时间优先,5、G1(Garbage First,jdk9默认),6、Full GC垃圾回收算法 1.标记清除 2.标记复制 3.标记整理 内存效率:复制算法>标记清除算法>标记压缩算法(时间复杂度) 内存整齐度:复制算法=标记清除算法>标记压缩算法 内存利用率:复制算法<标记清除算法=标记压缩算法 年轻代: 存活率低 复制算法 老年代: 区域大,存活率高 标记清除(内存碎片不是太多)+标记压缩共同实现 1、
主要内容:垃圾回收算法,1、垃圾回收器的分类,2、串行垃圾回收器,3、吞吐量优先,4、响应时间优先,5、G1(Garbage First,jdk9默认),6、Full GC垃圾回收算法 1.标记清除 2.标记复制 3.标记整理 内存效率:复制算法>标记清除算法>标记压缩算法(时间复杂度) 内存整齐度:复制算法=标记清除算法>标记压缩算法 内存利用率:复制算法<标记清除算法=标记压缩算法 年轻代: 存活率低 复制算法 老年代: 区域大,存活率高 标记清除(内存碎片不是太多)+标记压缩共同实现 1、