Java 执行GC判断对象是否存活有两种方式其中一种是引用计数
引用计数概念:JAVA Heap(堆)中每个对象有一个引用计数属性,新增一个引用时计数加1,引用释放时计数减1,计数为0时,GC垃圾回收会回收这些对象。引用分为很多种:强引用、软引用、弱引用、虚引用,下面会对比下每个引用之前的区别,和gc时候的生命周期。
强引用(StrongReference)如果一个对象具有强引用,那垃圾回收器绝不会回收它。
Object o = new Object(); // 强引用
只要强引用存在,对象就不会被gc掉。当然jvm停止(假死)或终止(kill掉)强引用会消失。
软引用(SoftReference)
如果一个对象只具有软引用,当jvm内存空间足够时,垃圾回收器就不会回收它;
如果内存空间不足了,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就可以被程序使用。软引用可用来实现内存式缓存。
SoftReference的构造器
软引用的GC,当JVM Heap内存不足的时候,在抛出OOM前会实行一次System.gc();处理掉软引用。
If(JVM.内存不足()) {
str = null; // 转换为软引用
System.gc(); // 垃圾回收器进行回收
}
弱引用(WeakReference):
看过ThreadLocalMap源码的同学应该会知道,Entry是继承了WeakReference,为了方便gc掉Entry中的key(key存放的类型是Threadlocal)。运用的就是弱引用原理,弱引用所关联的对象只能生存到下一次垃圾收集发生之前。此处附带一张ThreadLocalMap源码
一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。
虚引用(PhantomReference)
“虚引用”顾名思义,就是形同虚设,与其他几种引用都不同,虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。
ReferenceQueue queue = new ReferenceQueue ();
PhantomReference pr = new PhantomReference (object, queue);