Java

Java中的弱引用和软引用

2018-11-10  本文已影响9人  三年级一班亚索

参考:
https://blog.csdn.net/lqw_student/article/details/52947125

弱引用

对于只有弱引用的对象而言,当系统垃圾回收机制运行时,不管系统内存是否足够,总会回收该对象所占用的内存(立即回收的方式)。当然并不是说当一个对象只有弱引用时,它就会立即被回收—-正如那些使用引用的对象一样,必须等到系统垃圾回收机制运行时才会被回收。

public class Main {
    public static void main(String[] args) {
        String str = new String("Struts2权威指南");
        //创建一个弱引用,让这个弱引用引用到“Struts2权威指南”字符串
        WeakReference weakReference = new WeakReference(str);
        //切断str引用和“Struts2权威指南”字符串之间的引用
        str = null;
        //取出弱引用所引用的对象
        System.out.println(weakReference.get());
        //强制进行垃圾回收
        System.gc();
        System.runFinalization();
        //再次取出弱引用所引用的对象
        System.out.println(weakReference.get());
    }
}

运行结果:
“Struts2 权威指南”和null

软引用

软引用需要通过SoftReference类来实现,当一个对象只具有软引用时,它有可能被垃圾回收机制回收。对于只有软引用的对象而言,当系统内存空间足够时,它不会被系统回收,程序也可使用该对象;当系统内存空间不足时,系统将会回收它。软引用通常用于对内存敏感的程序中。

 public class Main {
        public static void main(String[] args) {
            String str = new String("Struts2权威指南");
            //创建一个软引用,让这个弱引用引用到“Struts权威指南”字符串
            SoftReference softReference = new SoftReference(str);
            //切断str引用和“Struts2权威指南”字符串之间的引用
            str = null; 
            //取出软引用所引用的对象
            System.out.println(softReference.get());
            //强制进行垃圾回收
            System.gc();
            System.runFinalization();
            //再次取出软引用所引用的对象
            System.out.println(softReference.get());
        }
    }

输出结果是:“Struts2 权威指南”和“Struts2权威指南”

虚引用

public class Main2 {
    public static void main(String[] args) {
           //创建一个字符串引用
        String str = new String("Struts2权威指南");
        //创建一个引用队列
        ReferenceQueue referenceQueue = new ReferenceQueue();
        //创建一个虚引用 让次虚引用引用到“Struts2权威指南”字符串
        PhantomReference phantomReference = new PhantomReference(str, referenceQueue);
        //切断str和“Struts2权威指南”之间的引用关系
        str = null;
        //取出虚引用所引用的对象,并不能通过虚引用访问被引用的对象,
        //所以此处输出的应该是null
        System.out.println(phantomReference.get());
        //强制进行垃圾回收
        System.gc();
        System.runFinalization();
        //取出引用队列最先进入队列中的引用于phantomReference进行比较
        System.out.println(referenceQueue.poll() == phantomReference);
    }
}

因为系统无法通过虚引用来获取被引用的对象,所以在执行第一个System.out.println(phantomReference.get())的时候,打印出来的是空值(即使此时系统并未进行强制垃圾回收)。当程序强制垃圾回收后,只有虚引用引用的字符串对象将会被垃圾回收,当被引用的对象被回收后,对象的引用将被添加到关联的引用队列中去(也就时说,虚引用指向的字符串对象会被垃圾回收器回收,而自己本身将被添加到与之关联的引用队列中去),因此我们才在第二个System.out.println(referenceQueue.poll() == phantomReference)中看到输出的true

引用队列

上一篇 下一篇

猜你喜欢

热点阅读