Java弱引用(WeakReference)的理解與使用 java
看到篇帖子, 國外一個技術面試官在面試senior java developer的時候, 問到一個weak reference相關的問題. 他沒有指望有人可以完整解釋清楚weak reference是什麼, 怎麼用, 只是指望有人可以提到這個concept和java的GC相關. 很惋惜的是, 20多個擁有5年以上java開發經驗的面試者中, 只有兩人知道weak reference的存在, 而其中只有一人實際用到過他. 無疑, 在interviewer眼中, 對於weak reference的理解和應用在面試中給了這一個interviewee至關多的加分. 因此, 將我對於這個技術的理解和使用總結在這篇博客裏, 但願讀者和本身經過讀和寫這篇帖子, 可以在之後的工做和麪試中得到加分.程序員
在Java裏, 當一個對象o被建立時, 它被放在Heap裏. 當GC運行的時候, 若是發現沒有任何引用指向o, o就會被回收以騰出內存空間. 或者換句話說, 一個對象被回收, 必須知足兩個條件: 1)沒有任何引用指向它 2)GC被運行.面試
在現實狀況寫代碼的時候, 咱們每每經過把全部指向某個對象的referece置空來保證這個對象在下次GC運行的時候被回收 (能夠用java -verbose:gc來觀察gc的行爲)函數
1 Object c = new Car(); 2 c=null;
可是, 手動置空對象對於程序員來講, 是一件繁瑣且違背自動回收的理念的. 對於簡單的狀況, 手動置空是不須要程序員來作的, 由於在java中, 對於簡單對象, 當調用它的方法執行完畢後, 指向它的引用會被從stack中popup, 因此他就能在下一次GC執行時被回收了.oop
可是, 也有特殊例外. 當使用cache的時候, 因爲cache的對象正是程序運行須要的, 那麼只要程序正在運行, cache中的引用就不會被GC給(或者說, cache中的reference擁有了和主程序同樣的life cycle). 那麼隨着cache中的reference愈來愈多, GC沒法回收的object也愈來愈多, 沒法被自動回收. 當這些object須要被回收時, 回收這些object的任務只有交給程序編寫者了. 然而這卻違背了GC的本質(自動回收能夠回收的objects).優化
因此, java中引入了weak reference. 相對於前面舉例中的strong reference:this
1 Object c = new Car(); //只要c還指向car object, car object就不會被回收
當一個對象僅僅被weak reference指向, 而沒有任何其餘strong reference指向的時候, 若是GC運行, 那麼這個對象就會被回收. weak reference的語法是:spa
1 WeakReference<Car> weakCar = new WeakReference(Car)(car);
當要得到weak reference引用的object時, 首先須要判斷它是否已經被回收:.net
1 weakCar.get();
若是此方法爲空, 那麼說明weakCar指向的對象已經被回收了.code
下面來看一個例子:
1 package weakreference; 2 /** 3 * @author wison 4 */ 5 public class Car { 6 private double price; 7 private String colour; 8 9 public Car(double price, String colour){ 10 this.price = price; 11 this.colour = colour; 12 } 13 14 public double getPrice() { 15 return price; 16 } 17 public void setPrice(double price) { 18 this.price = price; 19 } 20 public String getColour() { 21 return colour; 22 } 23 public void setColour(String colour) { 24 this.colour = colour; 25 } 26 27 public String toString(){ 28 return colour +"car costs $"+price; 29 } 30 31 }
在上例中, 程序運行一段時間後, 程序打印出"Object has been collected." 說明, weak reference指向的對象的被回收了.
值得注意的一點 , 即便有 car 引用指向對象, 且 car 是一個strong reference, weak reference weakCar指向的對象仍然被回收了. 這是由於java的編譯器在發現進入while循環以後, car 已經沒有被使用了, 因此進行了優化(將其置空?). 當把TestWeakReference.java修改成:
1 package weakreference; 2 3 import java.lang.ref.WeakReference; 4 5 /** 6 * @author wison 7 */ 8 public class TestWeakReference { 9 10 public static void main(String[] args) { 11 12 Car car = new Car(22000,"silver"); 13 WeakReference<Car> weakCar = new WeakReference<Car>(car); 14 15 int i=0; 16 17 while(true){ 18 System.out.println("here is the strong reference 'car' "+car); 19 if(weakCar.get()!=null){ 20 i++; 21 System.out.println("Object is alive for "+i+" loops - "+weakCar); 22 }else{ 23 System.out.println("Object has been collected."); 24 break; 25 } 26 } 27 } 28 29 }
weak reference指向的object就不會被回收了. 由於還有一個strong reference car 指向它.
* WeakReference的一個特色是它什麼時候被回收是不可肯定的, 由於這是由GC運行的不肯定性所肯定的. 因此, 通常用weak reference引用的對象是有價值被cache, 並且很容易被從新被構建, 且很消耗內存的對象.
在weak reference指向的對象被回收後, weak reference自己其實也就沒有用了. java提供了一個ReferenceQueue來保存這些所指向的對象已經被回收的reference. 用法是在定義WeakReference的時候將一個ReferenceQueue的對象做爲參數傳入構造函數.
-SoftReference
soft reference和weak reference同樣, 但被GC回收的時候須要多一個條件: 當系統內存不足時(GC是如何斷定系統內存不足? 是否有參數能夠配置這個threshold?), soft reference指向的object纔會被回收. 正由於有這個特性, soft reference比weak reference更加適合作cache objects的reference. 由於它能夠儘量的retain cached objects, 減小重建他們所需的時間和消耗.