执行GC的线程属于VM Thread,不是Java Thread。
什么时候触发GC?
(1)当剩余Eden空间不够装下新创建的对象实例时触发MinorGC;
(2)当老年代剩余空间不够装下直接丢进老年代或者到达晋升年龄的对象时触发FullGC;
触发GC后,需要挂起所有的Java Thread,来执行GC Roots的可达性分析算法获取可达对象和不可达对象。
如何挂起所有的Java Thread?
在JVM发起GC时,可以先忽略处于sleep或者blocked的Java线程,因为这些Java线程在进入sleep或者blocked状态时已经标记自己进入了Safe Region,在离开sleep或者blocked状态时,会检查系统是否已经完成了GC,如果完成了就继续执行,如果没有就继续等待,直到收到GC完成的信号为止。
然后通过设置安全点,可以在不太长的时间内让所有正在运行的Java线程遇到安全点,顺利地被挂起。
最后,开始GC。
-- 设置安全点
保证正在执行的线程在不太长的时间内会遇到GC的safe point;
缺点:对于当前没有在执行的线程,比如处于sleep或者blocked的线程,就不是很有效
-- 设置安全区域
基本想法:当一个线程因sleep或者处于blocked而没有在执行时,其对应的代码片段中变量及对象的引用关系是不会发生变化的,所以只需要在一个线程在进入sleep或者blocked时,标识自己进入了Safe Region,从而在一个线程sleep或者blocked期间,JVM发生GC了,
问题: