可达性分析是什么?
可达性分析是用来判断对象是否存活,通过"GC Roots"作为起点,从这个节点往下搜索,如果有有引用,则这个对象是存活的,如果没有则判定可回收的对象。
JVM可达性分析解决了什么问题?
可达性分析解决了引用计数法(reference counting)导致的循环引用对象及繁琐更新操作需要额外的空间问题。
JVM可达性分析是怎么GC的?
当对象第一次JVM GC扫到的时候,这时候,JVM缓存还不是立即将该对象GC掉,而是打上标记,并且把该对象放到待清除队列中(F-Queue),这时候给该对象再一次机会,若下次GC的时候发现该对象还是与GC Roots没有任何连接,则直接被GC掉,否则将从待清除队列中清除。
个人理解:
像死刑犯,虽然犯的罪很严重,一审只审了死刑,但还不是终审,如果下一审直接执行,那就GG了,如果下一审缓期几年执行,只要在这几年没有犯罪,没有被查出来还有其他罪,基本就是可以一直活下去。
实现源码
/**
*参考:深入理解Java JVM
* 功能描述: 此代码演示了两点:
* * 1.对象可以在被GC时自我拯救。
* * 2.这种自救的机会只有一次,因为一个对象的finalize()方法最多只会被系统自动调用一次
*
* @param:
* @return:
* @auther: csh
* @date: 2020/12/4 16:36
*/
public class FinalizeEscapeGC {
public static FinalizeEscapeGC SAVE_HOOK = null;
public void isAlive() {
System.out.println("yes, i am still alive :)");
}
@Override
protected void finalize() throws Throwable {
super.finalize();
System.out.println("finalize method executed!");
FinalizeEscapeGC.SAVE_HOOK = this;
}
public static void main(String[] args) throws Throwable {
SAVE_HOOK = new FinalizeEscapeGC();
//对象第一次成功拯救自己
SAVE_HOOK = null;
System.gc();
// 因为Finalizer方法优先级很低,暂停0.5秒,以等待它
Thread.sleep(500);
if (SAVE_HOOK != null) {
SAVE_HOOK.isAlive();
} else {
System.out.println("no, i am dead :(");
}
// 下面这段代码与上面的完全相同,但是这次自救却失败了
SAVE_HOOK = null;
System.gc();
// 因为Finalizer方法优先级很低,暂停0.5秒,以等待它
Thread.sleep(500);
if (SAVE_HOOK != null) {
SAVE_HOOK.isAlive();
} else {
System.out.println("no, i am dead :(");
}
}
}
java垃圾回收默认就是可达性标记对象是否存活。
参考文章:
https://zhuanlan.zhihu.com/p/109262576
https://www.jianshu.com/p/c79c5e02ebe6
https://blog.csdn.net/iter_zc/article/details/41892567