强引用,软引用,弱引用,虚引用。

用了Java怎么长时间一直不知道原来Java还有四种引用类型,这个引用类型和我们平常说的可不一样。这里的引用类型不是指数据类型的一种,而是指Java中的引用所分的四种类型。他们代表了JVM回收内存的四种强度,分别如下。

强引用:

Java中的引用,有点像C++的指针。通过引用,可以对堆中的对象进行操作。在某函数中,当创建了一个对象,该对象被分配在堆中,通过这个对象的引用才能对这个对象进行操作。

[java]  view plain  copy
  1. Object o=new Object();  

假设以上代码是在函数体内运行的,那么局部变量o将被分配在栈上,而对象实例,被分配在堆上。局部变量o指向Object实例所在的堆空间,通过o可以操作该实例,那么o就是Object的引用。


如果再创建一个赋值语句

[java]  view plain  copy
  1. Object oj=o;  
那么o指向的Object内存空间也会被oj所指向,此时我们可以说oj==o,那么ob.equals也肯定相等,即两个对象的值也肯定一样,当修改了o对象的时候,oj对象也会发生变化,或者当修改了oj对象的时候o对象也肯定会变化。所以如果想复制另外一个对象的值的话千万不要用这种方式,在多线程的情况下可能会有更槽糕的情况发生,你可以使用JDK自带的克隆方法,也可以重写克隆方法。

[java]  view plain  copy
  1. public class Test1 {  
  2.             public static void main(String[] args) {  
  3.                 student S=new student("1","我是大s");  
  4.                 student s=S;  
  5.                 System.out.println(s==S); //true  
  6.                 System.out.println(s.equals(S)); //true  
  7.                 System.out.println("打印出大S"+S); //id=1  
  8.                 System.out.println("打印出小s"+s); //id=1  
  9.                 S.setId("2");  
  10.                 System.out.println("修改后的大S"+S);  //id=2  
  11.                 System.out.println("未修改后的小s"+s); //id=2  
  12.                 s.setId("3");  
  13.                 System.out.println("未修改后的大S"+S); //id=3  
  14.                 System.out.println("修改后的小s"+s); //id=3  
  15.       
  16.             }  
  17. }  
  18. class student{  
  19.     String name;  
  20.     String id;  
  21.     public String getName() {  
  22.         return name;  
  23.     }  
  24.     public void setName(String name) {  
  25.         this.name = name;  
  26.     }  
  27.     public String getId() {  
  28.         return id;  
  29.     }  
  30.     public void setId(String id) {  
  31.         this.id = id;  
  32.     }  
  33.     public student(String name, String id) {  
  34.         super();  
  35.         this.name = name;  
  36.         this.id = id;  
  37.     }  
  38.     @Override  
  39.     public String toString() {  
  40.         return "student [name=" + name + ", id=" + id + "]";  
  41.     }  
  42.       
  43.       
  44. }  




上例中的大S和小S都是强引用,强引用具有如下特点:

1.强引用可以直接访问目标对象。

2.强引用所指向的对象在任何时候都不会被系统回收。JVM宁愿抛出OOM异常,也不会回收强引用所指向的对象。

3.强引用可能导致内存泄漏,为了避免内存泄漏,在使用完成之后我们可以把字符串对象设置为null,如果是集合的话可以使用

[java]  view plain  copy
  1. List list=new ArrayList<>();  
  2. list.clear();  

软引用:

软引用的强度是仅次于强引用的,如果一个对象只具有软引用,则内存空间足够,垃圾回收器就不会回收它;如果内存空间不足了,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就可以被程序使用。软引用可用来实现内存敏感的高速缓存。   

我们可以使用java.lang.ref.SoftReference来创建软引用;

[java]  view plain  copy
  1. String str=new String("abc");                                     // 强引用  
  2. SoftReference<String> softRef=new SoftReference<String>(str);     // 软引用  
当内存不足时,等价于:

[java]  view plain  copy
  1. If(JVM.内存不足()) {  
  2.    str = null;  // 转换为软引用  
  3.    System.gc(); // 垃圾回收器进行回收  
  4. }  

弱引用:

弱引用的强度比软引用更次,也就是说只具有弱引用的对象拥有更短暂的生命周期。在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。

如果这个对象是偶尔的使用,并且希望在使用时随时就能获取到,但又不想影响此对象的垃圾收集,那么你应该用 Weak Reference 来标记此对象。

[java]  view plain  copy
  1. String str=new String("abc");      
  2. WeakReference<String> abcWeakRef = new WeakReference<String>(str);  
  3. str=null;  
如果你想把这个对象变成强引用的话可以使用

[java]  view plain  copy
  1. String  abc = abcWeakRef.get();  

弱引用可以和一个引用队列(ReferenceQueue)联合使用,如果弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中。

当你想引用一个对象,但是这个对象有自己的生命周期,你不想介入这个对象的生命周期,这时候你就是用弱引用。

这个引用不会在对象的垃圾回收判断中产生任何附加的影响。

[java]  view plain  copy
  1. public class ReferenceTest {  
  2.   
  3.     private static ReferenceQueue<VeryBig> rq = new ReferenceQueue<VeryBig>();  
  4.   
  5.     public static void checkQueue() {  
  6.         Reference<? extends VeryBig> ref = null;  
  7.         while ((ref = rq.poll()) != null) {  
  8.             if (ref != null) {  
  9.                 System.out.println("In queue: " + ((VeryBigWeakReference) (ref)).id);  
  10.             }  
  11.         }  
  12.     }  
  13.   
  14.     public static void main(String args[]) {  
  15.         int size = 3;  
  16.         LinkedList<WeakReference<VeryBig>> weakList = new LinkedList<WeakReference<VeryBig>>();  
  17.         for (int i = 0; i < size; i++) {  
  18.             weakList.add(new VeryBigWeakReference(new VeryBig("Weak " + i), rq));  
  19.             System.out.println("Just created weak: " + weakList.getLast());  
  20.   
  21.         }  
  22.   
  23.         System.gc();   
  24.         try { // 下面休息几分钟,让上面的垃圾回收线程运行完成  
  25.             Thread.currentThread().sleep(6000);  
  26.         } catch (InterruptedException e) {  
  27.             e.printStackTrace();  
  28.         }  
  29.         checkQueue();  
  30.     }  
  31. }  
  32.   
  33. class VeryBig {  
  34.     public String id;  
  35.     // 占用空间,让线程进行回收  
  36.     byte[] b = new byte[2 * 1024];  
  37.   
  38.     public VeryBig(String id) {  
  39.         this.id = id;  
  40.     }  
  41.   
  42.     protected void finalize() {  
  43.         System.out.println("Finalizing VeryBig " + id);  
  44.     }  
  45. }  
  46.   
  47. class VeryBigWeakReference extends WeakReference<VeryBig> {  
  48.     public String id;  
  49.   
  50.     public VeryBigWeakReference(VeryBig big, ReferenceQueue<VeryBig> rq) {  
  51.         super(big, rq);  
  52.         this.id = big.id;  
  53.     }  
  54.   
  55.     protected void finalize() {  
  56.         System.out.println("Finalizing VeryBigWeakReference " + id);  
  57.     }  
  58. }  

虚引用

虚引用顾名思义就是形同虚设,虚引用也称为幻影引用:一个对象是都有虚引用的存在都不会对生存时间都构成影响,也无法通过虚引用来获取对一个对象的真实引用。唯一的用处:能在对象被GC时收到系统通知,JAVA中用PhantomReference来实现虚引用。

对比不同:


[java]  view plain  copy
  1. class Grocery {  
  2.         private static final int SIZE = 10000;  
  3.         // 属性d使得每个Grocery对象占用较多内存,有80K左右  
  4.         private double[] d = new double[SIZE];  
  5.         private String id;  
  6.         public Grocery(String id) {  
  7.             this.id = id;  
  8.         }  
  9.         public String toString() {  
  10.             return id;  
  11.         }  
  12.         public void finalize() {  
  13.             System.out.println("即将回收 " + id);  
  14.         }  
  15.     }  
  16.     public class References {  
  17.         private static ReferenceQueue<Grocery> rq = new ReferenceQueue<Grocery>();  
  18.         public static void checkQueue() {  
  19.             Reference<? extends Grocery> inq = rq.poll(); // 从队列中取出一个引用  
  20.             if (inq != null)  
  21.                 System.out.println("In queue: " + inq + " : " + inq.get());  
  22.         }  
  23.         public static void main(String[] args) {  
  24.             final int size = 10;  
  25.             // 创建10个Grocery对象以及10个软引用  
  26.             Set<SoftReference<Grocery>> sa = new HashSet<SoftReference<Grocery>>();  
  27.             for (int i = 0; i < size; i++) {  
  28.                 SoftReference<Grocery> ref = new SoftReference<Grocery>(  
  29.                         new Grocery("软引用 " + i), rq);  
  30.                 System.out.println("刚刚 创建了: " + ref.get());  
  31.                 sa.add(ref);  
  32.             }  
  33.             System.gc();  
  34.             checkQueue();  
  35.             // 创建10个Grocery对象以及10个弱引用  
  36.             Set<WeakReference<Grocery>> wa = new HashSet<WeakReference<Grocery>>();  
  37.             for (int i = 0; i < size; i++) {  
  38.                 WeakReference<Grocery> ref = new WeakReference<Grocery>(  
  39.                         new Grocery("弱引用 " + i), rq);  
  40.                 System.out.println("刚刚 创建了: " + ref.get());  
  41.                 wa.add(ref);  
  42.             }  
  43.             System.gc();  
  44.             checkQueue();  
  45.             // 创建10个Grocery对象以及10个虚引用  
  46.             Set<PhantomReference<Grocery>> pa = new HashSet<PhantomReference<Grocery>>();  
  47.             for (int i = 0; i < size; i++) {  
  48.                 PhantomReference<Grocery> ref = new PhantomReference<Grocery>(  
  49.                         new Grocery("abc " + i), rq);  
  50.                 System.out.println("刚刚 创建了: " + ref.get());  
  51.                 pa.add(ref);  
  52.             }  
  53.             System.gc();  
  54.             checkQueue();  
  55.         }  
  56. }  
在上面的这个例子中,依次创建了十个10个软引用、10个弱引用和10个虚引用,它们各自引用一个Grocery对象。并且在创建之后调用GC方法。从程序运行时的打印结果可以看出,虚引用形同虚设,它所引用的对象随时可能被垃圾回收,具有弱引用的对象拥有稍微长的生命周期,当垃圾回收器执行回收操作时,有可能被垃圾回收,具有软引用的对象拥有较长的生命周期,但在Java虚拟机认为内存不足的情况下,也会被垃圾回收。

总结

强引用:就像是老板(OOM)的亲儿子一样,在公司可以什么事都不干,但是千万不要老是占用公司的资源为他自己做事,记得用完公司的妹子之后,要让她们去工作(资源要懂得释放) 不然公司很可能会垮掉的。
软引用:有点像老板(OOM)的亲戚,在公司表现不好有可能会被开除,即使你投诉他(调用GC)上班看片,但是只要不被老板看到(被JVM检测到)就不会被开除(被虚拟机回收)。
弱引用:就是一个普通的员工,平常如果表现不佳会被开除(对象没有其他引用的情况下),遇到别人投诉(调用GC)上班看片,那开除是肯定了(被虚拟机回收)。
虚引用:这货估计就是个实习生跟临时工把,遇到事情的时候想到了你,没有事情的时候,秒秒钟拿出去顶锅,开除。

猜你喜欢

转载自blog.csdn.net/u010325193/article/details/80284444