Java WeakReference弱引用的理解與使用

前言: 看到篇帖子, 國(guó)外一個(gè)技術(shù)面試官在面試senior java developer的時(shí)候, 問到一個(gè)weak reference相關(guān)的問題. 他沒有期望有人能夠完整解釋清楚weak reference是什么, 怎么用, 只是期望有人能夠提到這個(gè)concept和java的GC相關(guān). 很可惜的是, 20多個(gè)擁有5年以上java開發(fā)經(jīng)驗(yàn)的面試者中, 只有兩人知道weak reference的存在, 而其中只有一人實(shí)際用到過他. 無疑, 在interviewer眼中, 對(duì)于weak reference的理解和應(yīng)用在面試中給了這一個(gè)interviewee相當(dāng)多的加分. 所以, 將我對(duì)于這個(gè)技術(shù)的理解和使用總結(jié)在這篇博客里, 希望讀者和自己通過讀和寫這篇帖子, 能夠在以后的工作和面試中獲得加分.

在Java里, 當(dāng)一個(gè)對(duì)象o被創(chuàng)建時(shí), 它被放在Heap里. 當(dāng)GC運(yùn)行的時(shí)候, 如果發(fā)現(xiàn)沒有任何引用指向o, o就會(huì)被回收以騰出內(nèi)存空間. 或者換句話說, 一個(gè)對(duì)象被回收, 必須滿足兩個(gè)條件: 1)沒有任何引用指向它 2)GC被運(yùn)行.

在現(xiàn)實(shí)情況寫代碼的時(shí)候, 我們往往通過把所有指向某個(gè)對(duì)象的referece置空來保證這個(gè)對(duì)象在下次GC運(yùn)行的時(shí)候被回收 (可以用java -verbose:gc來觀察gc的行為)

Object c = new Car();

c=null;

但是, 手動(dòng)置空對(duì)象對(duì)于程序員來說, 是一件繁瑣且違背自動(dòng)回收的理念的.??對(duì)于簡(jiǎn)單的情況, 手動(dòng)置空是不需要程序員來做的, 因?yàn)樵趈ava中, 對(duì)于簡(jiǎn)單對(duì)象, 當(dāng)調(diào)用它的方法執(zhí)行完畢后, 指向它的引用會(huì)被從stack中popup, 所以他就能在下一次GC執(zhí)行時(shí)被回收了.

但是, 也有特殊例外. 當(dāng)使用cache的時(shí)候, 由于cache的對(duì)象正是程序運(yùn)行需要的, 那么只要程序正在運(yùn)行, cache中的引用就不會(huì)被GC給(或者說, cache中的reference擁有了和主程序一樣的life cycle). 那么隨著cache中的reference越來越多, GC無法回收的object也越來越多, 無法被自動(dòng)回收. 當(dāng)這些object需要被回收時(shí), 回收這些object的任務(wù)只有交給程序編寫者了. 然而這卻違背了GC的本質(zhì)(自動(dòng)回收可以回收的objects).

所以, java中引入了weak reference. 相對(duì)于前面舉例中的strong reference:

Object c = new Car(); //只要c還指向car object, car object就不會(huì)被回收

?當(dāng)一個(gè)對(duì)象僅僅被weak reference指向, 而沒有任何其他strong reference指向的時(shí)候, 如果GC運(yùn)行, 那么這個(gè)對(duì)象就會(huì)被回收. weak reference的語法是:

WeakReference<Car> weakCar = new WeakReference<Car>(car);

?當(dāng)要獲得weak reference引用的object時(shí), 首先需要判斷它是否已經(jīng)被回收:

weakCar.get();

?如果此方法為空, 那么說明weakCar指向的對(duì)象已經(jīng)被回收了.

下面來看一個(gè)例子:

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;

}

}

}

}

在上例中, 程序運(yùn)行一段時(shí)間后, 程序打印出"Object has been collected." 說明, weak reference指向的對(duì)象的被回收了.

值得注意的一點(diǎn), 即使有car引用指向?qū)ο? 且car是一個(gè)strong reference, weak reference weakCar指向的對(duì)象仍然被回收了. 這是因?yàn)閖ava的編譯器在發(fā)現(xiàn)進(jìn)入while循環(huán)之后,car已經(jīng)沒有被使用了, 所以進(jìn)行了優(yōu)化(將其置空?). 當(dāng)把TestWeakReference.java修改為:

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){

System.out.println("here is the strong reference 'car' "+car);

if(weakCar.get()!=null){

i++;

System.out.println("Object is alive for "+i+" loops - "+weakCar);

}else{

System.out.println("Object has been collected.");

break;

}

}

}

}

weak reference指向的object就不會(huì)被回收了. 因?yàn)檫€有一個(gè)strong referencecar指向它.

* WeakReference的一個(gè)特點(diǎn)是它何時(shí)被回收是不可確定的, 因?yàn)檫@是由GC運(yùn)行的不確定性所確定的. 所以, 一般用weak reference引用的對(duì)象是有價(jià)值被cache, 而且很容易被重新被構(gòu)建, 且很消耗內(nèi)存的對(duì)象.

ReferenceQueue

在weak reference指向的對(duì)象被回收后, weak reference本身其實(shí)也就沒有用了. java提供了一個(gè)ReferenceQueue來保存這些所指向的對(duì)象已經(jīng)被回收的reference. 用法是在定義WeakReference的時(shí)候?qū)⒁粋€(gè)ReferenceQueue的對(duì)象作為參數(shù)傳入構(gòu)造函數(shù).

其他類型的references

-SoftReference

soft reference和weak reference一樣, 但被GC回收的時(shí)候需要多一個(gè)條件: 當(dāng)系統(tǒng)內(nèi)存不足時(shí)(GC是如何判定系統(tǒng)內(nèi)存不足? 是否有參數(shù)可以配置這個(gè)threshold?), soft reference指向的object才會(huì)被回收. 正因?yàn)橛羞@個(gè)特性, soft reference比weak reference更加適合做cache objects的reference. 因?yàn)樗梢员M可能的retain cached objects, 減少重建他們所需的時(shí)間和消耗.

-phantomReference

這個(gè)還沒想到應(yīng)用場(chǎng)景, 就先不說了. 有人在實(shí)踐中用到了的話, 歡迎分享.

WeakHashMap

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市提陶,隨后出現(xiàn)的幾起案子惧互,更是在濱河造成了極大的恐慌为流,老刑警劉巖扼雏,帶你破解...
    沈念sama閱讀 206,311評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件岂傲,死亡現(xiàn)場(chǎng)離奇詭異扫沼,居然都是意外死亡叁温,警方通過查閱死者的電腦和手機(jī)江掩,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門学辱,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人环形,你說我怎么就攤上這事策泣。” “怎么了抬吟?”我有些...
    開封第一講書人閱讀 152,671評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵萨咕,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我火本,道長(zhǎng)危队,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,252評(píng)論 1 279
  • 正文 為了忘掉前任钙畔,我火速辦了婚禮茫陆,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘擎析。我一直安慰自己簿盅,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,253評(píng)論 5 371
  • 文/花漫 我一把揭開白布叔锐。 她就那樣靜靜地躺著挪鹏,像睡著了一般。 火紅的嫁衣襯著肌膚如雪愉烙。 梳的紋絲不亂的頭發(fā)上讨盒,一...
    開封第一講書人閱讀 49,031評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音步责,去河邊找鬼返顺。 笑死,一個(gè)胖子當(dāng)著我的面吹牛蔓肯,可吹牛的內(nèi)容都是我干的遂鹊。 我是一名探鬼主播,決...
    沈念sama閱讀 38,340評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼蔗包,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼秉扑!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,973評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤舟陆,失蹤者是張志新(化名)和其女友劉穎误澳,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體秦躯,經(jīng)...
    沈念sama閱讀 43,466評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡忆谓,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,937評(píng)論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了踱承。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片倡缠。...
    茶點(diǎn)故事閱讀 38,039評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖茎活,靈堂內(nèi)的尸體忽然破棺而出昙沦,到底是詐尸還是另有隱情,我是刑警寧澤妙色,帶...
    沈念sama閱讀 33,701評(píng)論 4 323
  • 正文 年R本政府宣布桅滋,位于F島的核電站慧耍,受9級(jí)特大地震影響身辨,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜芍碧,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,254評(píng)論 3 307
  • 文/蒙蒙 一煌珊、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧泌豆,春花似錦定庵、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至贞远,卻和暖如春畴博,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背蓝仲。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工俱病, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人袱结。 一個(gè)月前我還...
    沈念sama閱讀 45,497評(píng)論 2 354
  • 正文 我出身青樓亮隙,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親垢夹。 傳聞我的和親對(duì)象是個(gè)殘疾皇子溢吻,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,786評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容