判断一个对象是否“已死”

在Java虚拟机中,对象可以通过new ,Class.newInstance,反射,克隆,反序列化创建对象,那么虚拟机又是如何判断对象是否已死呢?
简单来说就是失去引用,就是为null.
1.引用计数算法
给对象创建一个计数器,当对象被引用时,计数器+1,而失去引用时,计数器-1,当最后计数器为0的时候代表该对象失去引用
但是该算法所带来的问题:

public Object o = null;
public static void f1(){
	A a1= new A();
 	A a2= new A();
 	a1.o = a2;
 	a2.o = a1;
 	a1 = null;
 	a2 = null;
}
 

虽然a1,a2最后结果被赋了null,但是他们彼此还相互引用,计数为1,这是gc就无法回收他们.
2.通过可达性分析算法
以"GC Roots"为起点,开始向下搜索,途径的路径为引用链,如果一个对象没有在路径上则称这个对象不可达,就会被回收
可以作为GC Roots对象的有:虚拟机栈中引用的对象,方法去中类静态属性引用的对象,方法区中常量引用的对象,Native方法中引用的对象
总的来说就是看当前对象的引用状态,说到引用状态又分为四种(由强到弱):
1.强引用
直接new出来的,只要强引用还存在,那么就永远不会被gc回收。
2.软引用
指有用但是不是必须的对象,这些对象在当虚拟机抛出OOM时被gc回收
3.弱引用
描述非必须对象的,若引用的存活时间就是等到下一次进行gc时,gc将其回收
4.虚引用
完全不会对其生存空间构成影响,它的作用就是当这个对象被收集器回收时收到一个系统通知

说到这里其实仅仅是完成了判断的一半条件,另一半条件是执行finalize()方法,但是条件是当前对象之前没有调用过finalize()方法,那么这时,该对象会被放去F-Queue的队列中,并在稍后由虚拟机自动建立的低优先级的Finalize线程去执行。
注意!在Finalize线程还没有回收到当前对象前,该对象是有逃脱死亡的机会的,那就是重新引用!

JVM 判断对象是否存活主要有两种方法,分别是引用计数算法和可达性分析算法。 ### 引用计数算法 在对象中添加一个引用计数器,每有一个地方引用它,计数器的值加 1;当引用失效时,计数器的值减 1。任何时刻计数器为 0 的对象就是不再被使用的对象。不过,该算法很难解决对象之间相互循环引用的问题,可能会导致内存泄漏。例如以下 Java 代码: ```java public class ReferenceCountingTest { private Object instance; public ReferenceCountingTest() { byte[] b = new byte[20 * 1024 * 1024]; } public static void main(String[] args) { ReferenceCountingTest m1 = new ReferenceCountingTest(); ReferenceCountingTest m2 = new ReferenceCountingTest(); m1.instance = m2; m2.instance = m1; m1 = null; m2 = null; System.gc(); } } ``` 在这个例子中,`m1` 和 `m2` 相互引用,即使栈对它们的引用断开,由于计数器不为 0,引用计数算法无法回收它们[^1][^4]。 ### 可达性分析算法 这是主流商用程序语言(如 Java、C#)判断对象是否存活的常用方法。其基本思路是以一系列称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链。当一个对象到 GC Roots 没有任何引用链相连时,说明此对象是不可用的,将会被判定为可回收的对象。在 Java 语言中,能被认为是 GC Roots 的对象包括以下几种: 1. 虚拟机栈(栈帧中的本地变量表)中的引用的对象; 2. 本地方法栈中 JNI(Native 方法)引用的对象; 3. 方法区中类静态属性引用的对象; 4. 方法区中常量引用的对象; 5. 所有被同步锁(synchronized 关键字)持有的对象[^2][^3][^4][^5]。 要真正宣告一个对象亡,至少要经历两次标记过程。如果对象在进行可达性分析后发现没有与 GC Roots 相连接的引用链,那它将会被第一次标记,随后进行一次筛选,筛选的条件是此对象是否有必要执行 `finalize()` 方法。若有必要执行(即重写该方法),则筛选出来进入下一阶段(第二次标记和筛选);若没有必要(没有 `finalize()` 方法或者方法已经被执行过),则判断对象亡。第二次标记时,对象会被放在一个 F-queue 队列中,并由虚拟机自动建立优先级较低的 Finalizer 线程去执行该对象中的 `finalize()` 方法。`finalize()` 方法只会被执行一次,实际上并不承诺完成整个方法(防止方法执行缓慢或停止导致队列其他对象永久等待)。筛选标准是在执行 `finalize()` 方法后,如果该对象仍然没有跟 GC Roots 发生关联,就留在“即将回收”集合中等待回收[^2][^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值