從JDK1.2版本開始,把對象的引用分為四種級別,從而使程序能更加靈活的控制對象的生命周期。這四種級別由高到低依次為:強引用、軟引用、弱引用和虛引用。
強引用
這是使用最普遍的引用。如果一個對象具有強引用,那就類似于必不可少的生活用品,垃圾回收器絕不會回收它。當內存空 間不足,Java虛擬機寧愿拋出OutOfMemoryError錯誤,使程序異常終止,也不會靠隨意回收具有強引用的對象來解決內存不足問題,對于普通的User user = new User();的這種對象就是強引用。
軟引用
如果一個對象只具有軟引用,那就類似于可有可無的生活用品。如果內存空間足夠,垃圾回收器就不會回收它,如果內存空間不足了,就會回收這些對象的內存。只要垃圾回收器沒有回收它,該對象就可以被程序使用。軟引用可用來實現內存敏感的高速緩存。
軟引用可以和一個引用隊列(ReferenceQueue)聯合使用,如果軟引用所引用的對象被垃圾回收,JAVA虛擬機就會把這個軟引用加入到與之關聯的引用隊列中。
SoftReference引用的裝飾的對象,該對象的強引用應該賦值空。這里配合ReferenceQueue使用(也可以不配合),因為SoftReference本身也是new了一個對象。當他把別人的問題解決掉以后,也需要把自己的問題解決,所以當jvm把軟引用對象回收后,就會把自己的對象引用放到這個隊列中,我們可以通過隊列的poll()方法查看,如果存在該ReferenceQueue,那么就說明該對象的軟引用對象已經被jvm回收,我們需要將該ReferenceQueue引用也賦值為空,等待被jvm回收。
注意:軟引用對象是在jvm內存不夠的時候才會被回收,我們調用System.gc()方法只是起通知作用,jvm什么時候掃描回收對象是,是jvm自己的狀態決定的。就算掃描到軟引用對象也不一定會回收它。只有內存不夠的時候才會回收
ReferenceQueue queue = new ReferenceQueue();
User user = new User();
SoftReference ref=new SoftReference(user, queue);
user=null;
弱引用
如果一個對象只具有弱引用,那就類似于可有可物的生活用品。弱引用與軟引用的區別在于:只具有弱引用的對象擁有更短暫的生命周期。在垃圾回收器線程掃描它 所管轄的內存區域的過程中,一旦發現了只具有弱引用的對象,不管當前內存空間足夠與否,都會回收它的內存。不過,由于垃圾回收器是一個優先級很低的線程, 因此不一定會很快發現那些只具有弱引用的對象。
弱引用可以和一個引用隊列(ReferenceQueue)聯合使用,如果弱引用所引用的對象被垃圾回收,Java虛擬機就會把這個弱引用加入到與之關聯的引用隊列中。
弱引用也是用來描述非必需對象的,當JVM進行垃圾回收時,只要掃描到,
無論內存是否充足(與軟引用的區別),都會回收被弱引用關聯的對象。在java中,
用java.lang.ref.WeakReference類來表示
WeakReference sr = new WeakReference(new User());
System.out.println(sr.get());
System.gc(); //通知JVM的gc進行垃圾回收
System.out.println(sr.get());
虛引用
"虛引用"顧名思義,就是形同虛設,與其他幾種引用都不同,虛引用并不會決定對象的生命周期。如果一個對象僅持有虛引用,那么它就和沒有任何引用一樣,在任何時候都可能被垃圾回收。
虛引用主要用來跟蹤對象被垃圾回收的活動。虛引用與軟引用和弱引用的一個區別在于:虛引用必須和引用隊列(ReferenceQueue)聯合使用。當垃 圾回收器準備回收一個對象時,如果發現它還有虛引用,就會在回收對象的內存之前,把這個虛引用加入到與之關聯的引用隊列中。程序可以通過判斷引用隊列中是 否已經加入了虛引用,來了解
被引用的對象是否將要被垃圾回收。程序如果發現某個虛引用已經被加入到引用隊列,那么就可以在所引用的對象的內存被回收之前采取必要的行動。
虛引用和前面的軟引用、弱引用不同,它并不影響對象的生命周期。
在java中用java.lang.ref.PhantomReference類表示。如果一個對象與虛引用關聯,
則跟沒有引用與之關聯一樣,在任何時候都可能被垃圾回收器回收。
要注意的是,虛引用必須和引用隊列關聯使用,當垃圾回收器準備回收一個對象時,
如果發現它還有虛引用,就會把這個虛引用加入到與之 關聯的引用隊列中。
程序可以通過判斷引用隊列中是否已經加入了虛引用,來了解被引用的對象是否將要被垃圾回收。
如果程序發現某個虛引用已經被加入到引用隊列,那么就可以在所引用的對象的內存被回收之前采取必要的行動。
ReferenceQueue queue = new ReferenceQueue();
PhantomReference pr = new PhantomReference(new User(), queue);
System.out.println(pr.get());