*第一次原创,纯手打,不好请见谅。*
在学习ThreadLocal的时候,看到源码中提到了WeakReference,好奇心来了,在网上找到了一篇文章,看了半天,发现问题,所以在此记录下当时的经验。
WeakReference,即弱引用,我们都知道,在jvm中,对像是被存放在heap中的,而java提供了gc机制,目的在于不需要解放程序员的双手(不需要手动释放内存,避免了C++中的内存泄漏问题)。但是,被回收的两个条件是:1.gc在运行,2.对像上没有引用了。诚然,在大部分情况下,当退出某个方法时,stack会自动弹出,并销毁所有的引用,那么那些对象就可以被回收了。并不是所有的方法都会被立即执行完成啊。比如,可是缓存啊。
问题来了!怎么半,有人说,我手动把引用置空呗,有道理,那和 C++ 有什么区别?
解决方法就是:WeakReference。用法,WeakReference<T>,需要调用get方法,并且需要判断是否为null,如果为null,意味着此对象已被回收。
package weakreference;
/** * @author wison */
public class Car {
private double price;
private String colour;
public Car(double price, String colour){
this.price = price;
this.colour = colour;
}
public double getPrice() {
return price;
}
public void setPrice(double price) {
this.price = price;
}
public String getColour() {
return colour;
}
public void setColour(String colour) {
this.colour = colour;
}
public String toString(){
return colour +"car costs $"+price;
}
}
package weakreference;
import java.lang.ref.WeakReference;
/** * @author wison */
public class TestWeakReference {
public static void main(String[] args) {
Car car = new Car(22000,"silver");
WeakReference<Car> weakCar = new WeakReference<Car>(car);
int i=0;
while(true){
if(weakCar.get()!=null){
i++;
System.out.println("Object is alive for "+i+" loops - "+weakCar);
}else{
System.out.println("Object has been collected.");
break;
}
}
}
}
代码很简单,我就不解释了。
运行就可发现,最后真的被回收了。
等等,似乎有点问题?不是还有个强引用么?怎么也回收了啊?确实,请看Car car = new Car();怎么回事呢?
笔者猜测,是因为编译器优化了,因为下面的代码都没有用到这个引用,你可以试试在循环中,或者在循环外架上一个car的引用,自己试试吧。