Java的對(duì)象引用類型

強(qiáng)制系統(tǒng)垃圾回收的兩種方式:

調(diào)用System類的gc()靜態(tài)方法:System.gc()狐榔。
調(diào)用Runtime對(duì)象的gc()實(shí)例方法:Runtime.getRuntime.gc()坛增。

強(qiáng)制垃圾回收機(jī)制調(diào)用可恢復(fù)對(duì)象的finalize()方法以及通知系統(tǒng)進(jìn)行資源清理

調(diào)用System類的runFinalization()靜態(tài)方法:System.runFinalization()。
調(diào)用Runtime對(duì)象的runFinalization()實(shí)例方法:Runtime.getRuntime.runFinalization()薄腻。

Java對(duì)對(duì)象的4中引用方式

1收捣、強(qiáng)引用(StrongReference)

StrongReference是Java程序中最常見的引用方式。程序創(chuàng)建一個(gè)對(duì)象庵楷,并把這個(gè)對(duì)象賦給一個(gè)引用變量罢艾,程序通過(guò)該引用變量來(lái)操作實(shí)際的對(duì)象,當(dāng)一個(gè)對(duì)象被一個(gè)或一個(gè)以上的引用變量所引用時(shí)尽纽,它處于可達(dá)狀態(tài)咐蚯,不可能被系統(tǒng)垃圾回收機(jī)制回收。

@Test  
public void strongReference() {  
    Object referent = new Object();  
      
    /** 
     * 通過(guò)賦值創(chuàng)建 StrongReference  
     */  
    Object strongReference = referent;  
      
    assertSame(referent, strongReference);  
      
    referent = null;  
    System.gc();  
      
    /** 
     * StrongReference 在 GC 后不會(huì)被回收 
     */  
    assertNotNull(strongReference);  
} 

2弄贿、軟引用(SoftReference春锋,對(duì)應(yīng)JDK中java.lang.ref.SoftReference)

軟引用需要通過(guò)SoftReference類來(lái)實(shí)現(xiàn),當(dāng)一個(gè)對(duì)象只有軟引用時(shí)差凹,它有可能被垃圾回收機(jī)制回收看疙。對(duì)于只有軟引用的對(duì)象而言豆拨,當(dāng)系統(tǒng)內(nèi)存空間足夠時(shí),它不會(huì)被系統(tǒng)回收能庆,程序也可以使用該對(duì)象施禾;當(dāng)系統(tǒng)內(nèi)存空間不足時(shí),系統(tǒng)可能會(huì)回收它搁胆。軟引用通常用于對(duì)內(nèi)存敏感的程序中弥搞。

@Test  
public void softReference() {  
    Object referent = new Object();  
    SoftReference<Object> softRerference = new SoftReference<Object>(referent);  
  
    assertNotNull(softRerference.get());  
      
    referent = null;  
    System.gc();  
      
    /** 
     *  soft references 只有在 jvm OutOfMemory 之前才會(huì)被回收, 所以它非常適合緩存應(yīng)用 
     */  
    assertNotNull(softRerference.get());  
}  

3、WeakReference(弱引用渠旁,對(duì)應(yīng)JDK中java.lang.ref.WeakReference)

弱引用通過(guò)WeakReference類實(shí)現(xiàn)攀例,弱引用和軟引用很像,但弱引用的引用級(jí)別更低顾腊。對(duì)于只有弱引用的對(duì)象而言粤铭,當(dāng)系統(tǒng)回收機(jī)制運(yùn)行時(shí),不管系統(tǒng)內(nèi)存是否足夠杂靶,總會(huì)回收該對(duì)象占用的內(nèi)存梆惯。

@Test  
public void weakReference() {  
    Object referent = new Object();  
    WeakReference<Object> weakRerference = new WeakReference<Object>(referent);  
  
    assertSame(referent, weakRerference.get());  
      
    referent = null;  
    System.gc();  
      
    /** 
     * 一旦沒(méi)有指向 referent 的強(qiáng)引用, weak reference 在 GC 后會(huì)被自動(dòng)回收 
     */  
    assertNull(weakRerference.get());  
}  

4、WeakHashMap實(shí)現(xiàn)類

WeakHashMap與HashMap的用法基本相似吗垮。與HashMap的區(qū)別在于垛吗,HashMap的key保留了對(duì)實(shí)際的強(qiáng)引用,意味著該HashMap不會(huì)被銷毀烁登,其所有key所引用得對(duì)象就不會(huì)被回收怯屉,HashMap也不會(huì)自動(dòng)刪除這些key所對(duì)應(yīng)的key-value對(duì);但WeakHashMap的key只保留實(shí)際對(duì)象的弱引用饵沧,這些key所引用的對(duì)象可能被垃圾回收锨络,WeakHashMap也可能自動(dòng)刪除這些key對(duì)應(yīng)的key-value對(duì)。

public class WeakHashMapTest{
    public static void main(){
        WeakHashMap whm = new WeakHashMap();
        // 將WeakHashMap中添加三個(gè)key-value對(duì)狼牺,
        // 三個(gè)key都是匿名字符串對(duì)象(沒(méi)有其他引用)
        whm.put(new String("語(yǔ)文") , new String("良好"));
        whm.put(new String("數(shù)學(xué)") , new String("及格"));
        whm.put(new String("英文") , new String("中等"));
        //將 WeakHashMap中添加一個(gè)key-value對(duì)羡儿,
        // 該key是一個(gè)系統(tǒng)緩存的字符串對(duì)象。
        whm.put("java" , new String("中等"));    // ①
        // 輸出whm對(duì)象锁右,將看到4個(gè)key-value對(duì)。
        System.out.println(whm);
        // 通知系統(tǒng)立即進(jìn)行垃圾回收
        System.gc();
        System.runFinalization();
        // 通常情況下讶泰,將只看到一個(gè)key-value對(duì)咏瑟。
        System.out.println(whm);
    }
}

編譯、運(yùn)行上面程序痪署,看到如下運(yùn)行如果:

{英文=中等, java=中等, 數(shù)學(xué)=及格, 語(yǔ)文=良好}
{java=中等}

當(dāng)系統(tǒng)進(jìn)行垃圾回收時(shí)码泞,刪除了WeakHashMap對(duì)象的前三個(gè)key-value對(duì)。這是因?yàn)樘砑忧叭齻€(gè)key-value對(duì)(粗體字部分)時(shí)狼犯,這三個(gè)key都是匿名的字符串對(duì)象余寥,WeakHashMap只保留了對(duì)它們的弱引用领铐,這個(gè)垃圾回收時(shí)會(huì)自動(dòng)刪除這三個(gè)key-value對(duì)。
WeakHashMap對(duì)象中第4個(gè)組key-value對(duì)的key是一個(gè)字符串直接量宋舷,(系統(tǒng)會(huì)自動(dòng)保留對(duì)該字符串對(duì)象的強(qiáng)引用)绪撵,所以垃圾回收是不會(huì)回收它。

注:如果需要使用WeakHashMap的key來(lái)保留對(duì)象的弱引用祝蝠,則不要讓該key所引用的對(duì)象具有任何強(qiáng)引用音诈,否則將失去使用WeakHashMap的意義。

5绎狭、PhantomReference (虛引用细溅,對(duì)于JDK中java.lang.ref.PhantomReference )

通過(guò)PhantomReference類實(shí)現(xiàn),徐引用類似完全沒(méi)有引用儡嘶。虛引用對(duì)對(duì)象本身沒(méi)有太大影響喇聊,對(duì)象甚至感覺不到虛引用的存在。如果一個(gè)對(duì)象只有一個(gè)虛引用時(shí)蹦狂,它和沒(méi)有引用的效果大致相同誓篱。因?yàn)樗?get() 方法永遠(yuǎn)返回 null。

@Test  
public void phantomReferenceAlwaysNull() {  
    Object referent = new Object();  
    PhantomReference<Object> phantomReference = new PhantomReference<Object>(referent, new ReferenceQueue<Object>());  
      
    /** 
     * phantom reference 的 get 方法永遠(yuǎn)返回 null  
     */  
    assertNull(phantomReference.get());  
}

注:SoftReference鸥咖、WeakReference以及PhantomReference均包含一個(gè)get()方法燕鸽,用于獲取被它們所引用的對(duì)象。

6啼辣、ReferenceQueue(引用隊(duì)列啊研,對(duì)于JDK中java.lang.ref.ReferenceQueue)

用于保存被回收后對(duì)象的引用。當(dāng)聯(lián)合使用軟引用鸥拧、弱引用和引用隊(duì)列時(shí)党远,系統(tǒng)在回收被引用的對(duì)象之后,將把被回收對(duì)象對(duì)應(yīng)的引用添加到關(guān)聯(lián)的引用隊(duì)列中富弦。與軟引用和弱引用不同的是沟娱,虛引用在對(duì)象釋放之前,將把它對(duì)應(yīng)的虛引用添加到它關(guān)聯(lián)的引用隊(duì)列中腕柜,這使得可以在對(duì)象在被回收之前采取行動(dòng)济似。程序可以檢查與虛引用關(guān)聯(lián)的引用隊(duì)列中是否已經(jīng)包含了該虛引用,從而了解虛引用所引用的對(duì)象是否即將被回收盏缤。

public class PhantomReferenceTest
{
    public static void main(String[] args)
        throws Exception
    {
        // 創(chuàng)建一個(gè)字符串對(duì)象
        String str = new String("瘋狂Java講義");
        // 創(chuàng)建一個(gè)引用隊(duì)列
        ReferenceQueue rq = new ReferenceQueue();
        // 創(chuàng)建一個(gè)虛引用砰蠢,讓此虛引用引用到"瘋狂Java講義"字符串
        PhantomReference pr = new PhantomReference (str , rq);
        // 切斷str引用和"瘋狂Java講義"字符串之間的引用
        str = null;
        // 取出虛引用所引用的對(duì)象,并不能通過(guò)虛引用獲取被引用的對(duì)象唉铜,所以此處輸出null
        System.out.println(pr.get());  //  ①
        // 強(qiáng)制垃圾回收
        System.gc();
        System.runFinalization();
        // 垃圾回收之后台舱,虛引用將被放入引用隊(duì)列中
        // 取出引用隊(duì)列中最先進(jìn)入隊(duì)列中的引用與pr進(jìn)行比較
        System.out.println(rq.poll() == pr);   // ②
    }
}

因?yàn)橄到y(tǒng)無(wú)法通過(guò)虛引用獲得被引用的對(duì)象,多以執(zhí)行①處的輸出語(yǔ)句時(shí)潭流,程序輸出null竞惋。當(dāng)程序強(qiáng)制垃圾回收回收后柜去,只有虛引用引用的字符串對(duì)象將被垃圾回收,當(dāng)被引用的對(duì)象被回收后拆宛,對(duì)應(yīng)的虛引用將被添加到關(guān)聯(lián)的引用隊(duì)列中嗓奢,因而將在②代碼處看到輸出true。

由于垃圾回收的不確定性胰挑,當(dāng)程序希望從軟蔓罚、弱引用中獲取被引用對(duì)象時(shí),可能這個(gè)對(duì)象已經(jīng)被釋放了瞻颂。如果程序需要使用被引用對(duì)象豺谈,則必須被重新創(chuàng)建該對(duì)象。這個(gè)過(guò)程可以采用兩種方式:

// 取出弱引用所引用的對(duì)象
obj = wr.get();
// 如果取出的對(duì)象為null
if(obj == null){
    // 重新創(chuàng)建一個(gè)新的對(duì)象贡这,再次讓弱引用去引用該對(duì)象
    wr = new WeakReference(recreateIt()) ;
    // 取出弱引用所引用的對(duì)象茬末,將其賦給obj變量
    obj = wr.get() ;
    ... // 操作obj對(duì)象
    // 再次切斷obj和對(duì)象之間的關(guān)聯(lián)
}
// 再次切斷obj和對(duì)象之間的關(guān)聯(lián)
obj = null ;

recreateIt()表示創(chuàng)建一個(gè)obj對(duì)象,這段代碼存在一個(gè)問(wèn)題盖矫,就是垃圾回收的不確定性丽惭,可能會(huì)導(dǎo)致新創(chuàng)建的弱引用再次被回收,從而獲取到的obj仍然是null辈双。另一種取出方式避免了這個(gè)問(wèn)題:

// 取出弱引用所引用的對(duì)象
obj = wr.get();
// 如果取出的對(duì)象為null
if(obj == null){
    // 重新創(chuàng)建一個(gè)新的對(duì)象责掏,并使用強(qiáng)引用來(lái)引用它
    obj = recreateIt() ;
    // 取出弱引用所引用的對(duì)象,將其賦給obj變量
    wr = new WeakReference(obj) ;
    ... // 操作obj對(duì)象
    // 再次切斷obj和對(duì)象之間的關(guān)聯(lián)
}
// 再次切斷obj和對(duì)象之間的關(guān)聯(lián)
obj = null ;

7湃望、PhantomReference vs WeakReference

PhantomReference 有兩個(gè)好處:
1换衬、它可以讓我們準(zhǔn)確地知道對(duì)象何時(shí)被從內(nèi)存中刪除, 這個(gè)特性可以被用于一些特殊的需求中(例如 Distributed GC证芭, XWork 和 google-guice 中也使用 PhantomReference 做了一些清理性工作)瞳浦。
2、它可以避免 finalization 帶來(lái)的一些根本性問(wèn)題, 上文提到 PhantomReference 的唯一作用就是跟蹤 referent 何時(shí)被 enqueue 到 ReferenceQueue 中, 但是 WeakReference 也有對(duì)應(yīng)的功能, 兩者的區(qū)別到底在哪呢 ?
這就要說(shuō)到 Object 的 finalize 方法, 此方法將在 gc 執(zhí)行前被調(diào)用, 如果某個(gè)對(duì)象重載了 finalize 方法并故意在方法內(nèi)創(chuàng)建本身的強(qiáng)引用, 這將導(dǎo)致這一輪的 GC 無(wú)法回收這個(gè)對(duì)象并有可能引起任意次 GC废士, 最后的結(jié)果就是明明 JVM 內(nèi)有很多 Garbage 卻 OutOfMemory叫潦, 使用 PhantomReference 就可以避免這個(gè)問(wèn)題, 因?yàn)?PhantomReference 是在 finalize 方法執(zhí)行后回收的官硝,也就意味著此時(shí)已經(jīng)不可能拿到原來(lái)的引用, 也就不會(huì)出現(xiàn)上述問(wèn)題, 當(dāng)然這是一個(gè)很極端的例子, 一般不會(huì)出現(xiàn)矗蕊。

參考資料:《瘋狂Java講義(第3版)》
參考作者:舞熊科技 - 文章鏈接
轉(zhuǎn)發(fā)申明:我們不占有不侵權(quán),我們只是好文的搬運(yùn)工氢架!轉(zhuǎn)發(fā)請(qǐng)帶上原文申明傻咖。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市达箍,隨后出現(xiàn)的幾起案子没龙,更是在濱河造成了極大的恐慌铺厨,老刑警劉巖缎玫,帶你破解...
    沈念sama閱讀 222,464評(píng)論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件硬纤,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡赃磨,警方通過(guò)查閱死者的電腦和手機(jī)筝家,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,033評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)邻辉,“玉大人溪王,你說(shuō)我怎么就攤上這事值骇∮猓” “怎么了吱瘩?”我有些...
    開封第一講書人閱讀 169,078評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)使碾。 經(jīng)常有香客問(wèn)我蜜徽,道長(zhǎng),這世上最難降的妖魔是什么票摇? 我笑而不...
    開封第一講書人閱讀 59,979評(píng)論 1 299
  • 正文 為了忘掉前任,我火速辦了婚禮矢门,結(jié)果婚禮上盆色,老公的妹妹穿的比我還像新娘颅和。我一直安慰自己傅事,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,001評(píng)論 6 398
  • 文/花漫 我一把揭開白布峡扩。 她就那樣靜靜地躺著蹭越,像睡著了一般教届。 火紅的嫁衣襯著肌膚如雪响鹃。 梳的紋絲不亂的頭發(fā)上案训,一...
    開封第一講書人閱讀 52,584評(píng)論 1 312
  • 那天买置,我揣著相機(jī)與錄音强霎,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛轩触,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播脱柱,決...
    沈念sama閱讀 41,085評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼榨为!你這毒婦竟也來(lái)了惨好?” 一聲冷哼從身側(cè)響起随闺,我...
    開封第一講書人閱讀 40,023評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎矩乐,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體绰精,經(jīng)...
    沈念sama閱讀 46,555評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,626評(píng)論 3 342
  • 正文 我和宋清朗相戀三年笨使,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片硫椰。...
    茶點(diǎn)故事閱讀 40,769評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖靶草,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情奕翔,我是刑警寧澤,帶...
    沈念sama閱讀 36,439評(píng)論 5 351
  • 正文 年R本政府宣布派继,位于F島的核電站,受9級(jí)特大地震影響驾窟,放射性物質(zhì)發(fā)生泄漏庆猫。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,115評(píng)論 3 335
  • 文/蒙蒙 一绅络、第九天 我趴在偏房一處隱蔽的房頂上張望月培。 院中可真熱鬧嘁字,春花似錦、人聲如沸杉畜。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,601評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)寻行。三九已至,卻和暖如春匾荆,著一層夾襖步出監(jiān)牢的瞬間拌蜘,已是汗流浹背牙丽。 一陣腳步聲響...
    開封第一講書人閱讀 33,702評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留烤芦,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,191評(píng)論 3 378
  • 正文 我出身青樓构罗,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親遂唧。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,781評(píng)論 2 361

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

  • 1 Java的引用 對(duì)于Java中的垃圾回收機(jī)制來(lái)說(shuō)纹烹,對(duì)象是否被應(yīng)該回收的取決于該對(duì)象是否被引用。因此召边,引用也是J...
    高級(jí)java架構(gòu)師閱讀 395評(píng)論 0 1
  • 前言 從Jdk1.2開始宴卖,在java.lang.ref包下就提供了三個(gè)類:SoftReference(軟引用)邻悬,P...
    Aldeo閱讀 2,992評(píng)論 5 11
  • 最近觀察到很多同學(xué)著急忙慌地做事,但在很多最基本的工作上犯錯(cuò)父丰,持續(xù)犯錯(cuò)掘宪。準(zhǔn)備花點(diǎn)時(shí)間把工作過(guò)程里已經(jīng)形成肌肉記憶的...
    Lil_Sun閱讀 1,344評(píng)論 0 5
  • A攘烛;今天跟外部門溝通數(shù)據(jù)提供的準(zhǔn)確性問(wèn)題魏滚,前期提供的系統(tǒng)使用率數(shù)據(jù)一直不準(zhǔn)坟漱,上周反饋后今天跟進(jìn)結(jié)果 M:有點(diǎn)煩躁鼠次,...
    yx_yxyx閱讀 221評(píng)論 0 0
  • 窗臺(tái)上的沙漠玫瑰開花了芋齿。它已經(jīng)沉寂了大半年,我一度認(rèn)為它只是會(huì)一直這么綠著觅捆,與世無(wú)爭(zhēng)的樣子赦役,像個(gè)水蘿卜一樣存在而已...
    毒丸閱讀 224評(píng)論 0 0