通过这个link但仍然如此 混淆了小型和主要GC收集实际发生的事情。
假设我年轻一代有100个物体,其中85个物体是无法接触的物体。现在当Minor GC运行时, 它将回收85个物体的记忆并将15个物体移动到较旧(终身)的一代。
现在有15个活动对象存在于老一代中,其中3个无法访问。说主要GC发生。它会保持
15个对象,并回收3个无法访问的对象的内存。据说主要GC比次要GC慢。 My question is why ? Is it because of major GC happens on generally greater number of objects than minor as minor gc occurs more frequently than major?
根据理解major GC should be faster as it needs to do less work
,即从无法访问的对象中回收内存而不是次要GC,因为
年轻一代的高死亡率。
答案 0 :(得分:5)
1)次要GC将首先将15个物体移动到幸存者空间之一,例如SS1,下一个GC会将仍然活着的人移动到SS2,下一个GC会将幸存的人移回SS1,依此类推。只有那些幸存下来的几个(例如8个)重新安置(小型GCs)的人才能最终进入老一代。
2)仅当JVM无法在旧代中分配对象时才会发生主要GC,因为其中没有可用空间。为了清除死对象中的内存,GC遍历旧一代的所有对象,因为旧一代比新一代大几倍,它可能容纳数倍的对象,因此GC处理需要几倍的时间
答案 1 :(得分:2)
我的问题是为什么?是因为主要的GC发生在通常更多的对象上而不是次要的,因为较小的gc比主要的更频繁发生?
你几乎击中了它的头。从Oracle文章中强调我的:
主要集合通常要慢得多,因为它涉及所有实时对象。
因此,一个主要的GC不仅分析了旧一代的15个对象,而且还经历了年轻一代(再次)和permgen和GC这些堆的区域。次要GC只分析年轻一代,因此通常不会有多少对象可供查看。
根据理解,主要GC应该更快,因为它需要做更少的工作(即从无法到达的对象回收内存)而不是次要GC,因为年轻一代的死亡率很高。
我想我理解你为什么这么认为。我可以想象,在一个次要的GC之后很快就可以运行主要的GC,当时对象被提升到几乎完全老一代。因此,年轻一代(可能)不会包含太多要收集的物品。
然而,如果我记得正确的事情,老一代通常比年轻一代更大,所以GC不仅需要分析更多空间,还必须重新审视permgen,以及年轻一代的剩余物品(再次)。所以这可能就是为什么主要的GC速度较慢的原因 - 仅仅是因为还有更多的事情需要做。通过改变生成空间的大小,使年轻一代比老一代和permgen都大,你可能能够使主要GC比次要GC更快,但我不认为这将是一个常见的设置...