提問(wèn)Java四大引用潘悼?

四大引用各自回收條件?獲取對(duì)象方式爬橡?觸發(fā)OOM概率治唤?

回答這個(gè)問(wèn)題的最好方式就是編碼驗(yàn)證

本文代碼運(yùn)行環(huán)境:Mac os + jdk se 8 + VM options (-Xms10m -Xmx10m)

被測(cè)試對(duì)象類(lèi)RefTestObj

java-ref-RefTestObj.png

RefTestObj定義了三個(gè)成員變量糙申,重寫(xiě)了toString和finalize方法宾添。

測(cè)試OOM

測(cè)試代碼:使用不同引用類(lèi)型循環(huán)創(chuàng)建10w個(gè)RefTestObj對(duì)象,測(cè)試其發(fā)生OOM概率柜裸。

java-ref-refTestOOM.png
  • 強(qiáng)引用:調(diào)用 refTestOOM(0) 缕陕,大概在創(chuàng)建1.5w個(gè)對(duì)象時(shí)出現(xiàn)OOM,在此之前沒(méi)有打印過(guò)任何finalize方法日志疙挺。
  • 軟引用:調(diào)用 refTestOOM(1) 扛邑,順利創(chuàng)建完所有對(duì)象。 期間控制臺(tái)有幾次(大量對(duì)象的)finalize方法日志輸出铐然。
  • 弱引用:調(diào)用tefTestOOM(2) 蔬崩,順利創(chuàng)建完所有對(duì)象欢峰。 期間控制臺(tái)基本都在打印finalize方法日志铐刘。
  • 虛引用:調(diào)用refTestOOM(3) 几于,大概在創(chuàng)建3w個(gè)對(duì)象時(shí)出現(xiàn)OOM讳侨,在此之前有打印過(guò)finalize方法日志陷寝。
  • 細(xì)心的讀者可能已經(jīng)發(fā)現(xiàn)測(cè)試代碼中124行薄湿,做了100ms延時(shí)操作艇抠。如果沒(méi)有這個(gè)延時(shí)妒御,軟引用和弱引用也會(huì)出現(xiàn)OOM樟氢,原因請(qǐng)看代碼注釋和下一個(gè)問(wèn)題冈绊。
測(cè)試結(jié)果
引用類(lèi)型 回收條件 獲取對(duì)象方式 觸發(fā)OOM概率
強(qiáng)引用 不回收 通過(guò)引用
軟引用 內(nèi)存緊張時(shí) get方法
弱引用 GC時(shí) get方法 很低
虛引用 不確定? 無(wú)法獲取 較高埠啃?
  • StrongRef 直到OOM也不回收其引用的對(duì)象死宣。 通過(guò)引用直接獲取目標(biāo)對(duì)象實(shí)例。
  • SoftRef 內(nèi)存吃緊時(shí)回收其引用的對(duì)象碴开。 通過(guò)get方法獲取對(duì)象引用毅该,可能為null。 不會(huì)觸發(fā)OOM(如果沒(méi)時(shí)間釋放內(nèi)存也會(huì)觸發(fā)OOM(比如創(chuàng)建對(duì)象太快))
  • WeakRef GC時(shí)觸發(fā)回收潦牛。 get方法獲取對(duì)象引用眶掌,可能為null。 不會(huì)觸發(fā)OOM巴碗。(如果沒(méi)時(shí)間釋放內(nèi)存也會(huì)觸發(fā)OOM(比如創(chuàng)建對(duì)象太快)) 朴爬。 較SoftRef 對(duì)象執(zhí)行finalize更頻繁,極端情況出現(xiàn)OOM概率也更低橡淆。
  • PhantomRef 回收條件:不確定召噩,看日志有執(zhí)行finalize函數(shù)。 無(wú)法獲取對(duì)象實(shí)例逸爵。 會(huì)觸發(fā)OOM(較SoftRef和WeakRef概率高, 感覺(jué)和StrongRef差不多)具滴。

為什么使用了WeakRef(SoftRef)也會(huì)觸發(fā)OOM?

  • 創(chuàng)建對(duì)象時(shí)师倔,需要在堆中申請(qǐng)內(nèi)存抵蚊,如果申請(qǐng)時(shí)內(nèi)存不夠就會(huì)觸發(fā)OOM。
  • JVM根據(jù)算法在整理和釋放對(duì)象(沒(méi)有使用的)內(nèi)存溯革,而且這兩個(gè)過(guò)程不是同步的贞绳。

對(duì)象的finalize方法什么時(shí)候被執(zhí)行? 什么時(shí)候被放入ReferenceQueue中的致稀?

公用測(cè)試代碼 referenceQueueMonitor :用于監(jiān)控ReferenceQueue中對(duì)象冈闭,如果有則取出并打印日志。

java-ref-queueMonitor.png

測(cè)試弱引用 testWeakReferenceQueue

java-ref-testWeakReferenceQueue.png
運(yùn)行日志
java-ref-testWeakReferenceQueue-result.png
結(jié)果分析
  • 弱引用在GC觸發(fā)后就回收對(duì)象抖单。
  • GC后其指向的對(duì)象會(huì)被加入關(guān)聯(lián)的ReferenceQueue萎攒,和執(zhí)行對(duì)象的finalize方法遇八。
  • 關(guān)于finalize方法調(diào)用:1. 由JVM調(diào)用,時(shí)機(jī)不確定耍休, 2. 最多只調(diào)用一次刃永。

測(cè)試軟引用 testSoftReferenceQueue

java-ref-testSoftReferenceQueue.png
運(yùn)行日志
java-ref-testSoftReferenceQueue-result.png
結(jié)果分析
  • 從日志可以看到,通過(guò)19次gc后軟引用指向的對(duì)象終于被清理了羊精。(每多一次gc斯够,就會(huì)多創(chuàng)建1000個(gè)對(duì)象,以此來(lái)模擬內(nèi)存吃緊的情況喧锦。)
  • 軟引用指向?qū)ο髸?huì)在內(nèi)存接近滿負(fù)荷時(shí)读规,垃圾回收器將會(huì)清理該對(duì)象;可能還會(huì)根據(jù)SoftReference.get()方法的調(diào)用情況(比如很長(zhǎng)時(shí)間未調(diào)用)來(lái)決定要不要回收燃少。
  • 從日志來(lái)看軟引用在執(zhí)行對(duì)象的 finalize 方法之前被加入了 ReferenceQueue束亏;但實(shí)際上這兩個(gè)是相互獨(dú)立的邏輯,可參看下文 源碼分析 章節(jié)阵具。

測(cè)試虛引用 testPhantomReferenceQueue

java-ref-testPhantomReferenceQueue.png
運(yùn)行日志
java-ref-testPhantomReferenceQueue-result.png
結(jié)果分析
  • 虛引用在GC觸發(fā)后 , 直接調(diào)用了 finalize() 方法 , 但不會(huì)立即將其加入回收隊(duì)列 .
  • 只有在真正對(duì)象被 GC 清除時(shí) , 才會(huì)將其加入 ReferenceQueue 隊(duì)列中去 .
  • 對(duì)于虛引用對(duì)象碍遍,到底在經(jīng)過(guò)多次 GC 之后, 才會(huì)加入到隊(duì)列中去呢阳液? (經(jīng)過(guò)測(cè)試怕敬,發(fā)現(xiàn)在二次GC后會(huì)將其加入隊(duì)列;關(guān)于什么時(shí)機(jī)將對(duì)象加入隊(duì)列趁舀,各個(gè)虛擬機(jī)實(shí)現(xiàn)應(yīng)該是有差異的赖捌。

以上只是帶著問(wèn)題并通過(guò)編碼來(lái)驗(yàn)證的過(guò)程,如果想搞清楚答案為什么是這樣而不是那樣矮烹,就需要翻一翻jdk相關(guān)的代碼了越庇。

源碼實(shí)現(xiàn)

對(duì)象是怎么被加入ReferenceQueue的?

在java.lang.ref.Reference類(lèi)中定義有下圖所示的一段靜態(tài)代碼奉狈。

java-ref-reference-static-code.png

由上圖代碼可以看出卤唉,在Reference類(lèi)裝載時(shí)就會(huì)啟動(dòng)一個(gè)名叫ReferenceHandler的高優(yōu)先級(jí)的守護(hù)線程。ShareSecrets類(lèi)相關(guān)邏輯先忽略仁期,我們繼續(xù)看看ReferenceHandler線程類(lèi)的定義桑驱。

java-ref-reference-referenceHandler.png

重點(diǎn)關(guān)注run方法□说埃可以看到一個(gè)無(wú)限循環(huán)邏輯在執(zhí)行tryHandlePending(true)方法熬的,在看tryHandlePending方法代碼前,我們先看下Reference類(lèi)中定義的幾個(gè)變量赊级。(因?yàn)樗鼈兒苤匾嚎颍脖阌诤罄m(xù)代碼理解)

java-ref-reference-member-code.png

簡(jiǎn)單說(shuō)下,

  • 變量 referent -> 對(duì)象引用實(shí)例理逊。
  • 變量 queue -> 該引用類(lèi)型關(guān)聯(lián)的ReferenceQueue實(shí)例橡伞,它本身并不是一個(gè)隊(duì)列結(jié)構(gòu)實(shí)現(xiàn)盒揉。
  • 變量 next -> 對(duì)象入隊(duì)后用此變量來(lái)維護(hù)一個(gè)鏈表結(jié)構(gòu)。
  • 變量 discoveredpending 為Reference類(lèi)型兑徘,均有VM調(diào)用并賦值刚盈。
  • 變量 lock 同步鎖對(duì)象。我理解是VM檢測(cè)到有待處理對(duì)象時(shí)會(huì)通過(guò)此對(duì)象喚醒ReferenceHandler線程挂脑。

下面再來(lái)看看 tryHandlePending 方法實(shí)現(xiàn)藕漱。

java-ref-reference-tryHandlePending.png

方法主要邏輯:獲取對(duì)象鎖并判斷 pending 變量是否為null?

  • 為null:根據(jù)參數(shù)waitForNotify 決定是 wait 還是繼續(xù)循環(huán)執(zhí)行最域。
  • 不為null:進(jìn)行賦值谴分,并判斷如果不是Cleaner對(duì)象則將其放入隊(duì)列中锈麸。

到這里镀脂,關(guān)于Reference對(duì)象如何被放入隊(duì)列的代碼邏輯就分析完了。前面我有提到說(shuō)java.lang.ref.ReferenceQueue類(lèi)并不是一個(gè)隊(duì)列忘伞,那我們看看其 enqueue(r) 方法是如何實(shí)現(xiàn)的薄翅?

java-ref-referenceQueue-enqueue.png

其實(shí)它是使用自定義的一個(gè) head 變量和Reference類(lèi)的 next 變量來(lái)實(shí)現(xiàn)的隊(duì)列效果。

這里也有l(wèi)ock鎖對(duì)象氓奈,最后一句代碼還執(zhí)行了notifyAll()翘魄,原因就是該類(lèi)對(duì)外提供了一個(gè)阻塞 remove 方法,調(diào)用notifyAll()可喚醒在該鎖上等待的線程舀奶,具體代碼就不貼了暑竟,感興趣的讀者請(qǐng)自行查閱。

對(duì)象的 finalize()方法執(zhí)行邏輯育勺?

在java.lang.ref.Finalizer類(lèi)中定義有如下代碼但荤,可以看出FinalizerThread線程在類(lèi)裝載時(shí)就被啟動(dòng)了。

java-ref-Finalizer-FinalizerThread.png

FinalizerThread 線程主要邏輯就是從引用隊(duì)列 queue 中取出Finalizer對(duì)象涧至,并執(zhí)行其 runFinalizer()方法腹躁。

注意:這里的 queue 是Finalizer類(lèi)中定義的靜態(tài)變量和我們創(chuàng)建引用對(duì)象傳入的queue不是同一個(gè)哦。

繼續(xù)跟進(jìn) runFinalizer方法 ~

java-ref-Finalizer-runFinalizer.png

上圖中第 97 行代碼 remove 方法主要是將當(dāng)前Finalizer引用(對(duì)象)從鏈表中移除南蓬;第101 纺非、102 行代碼可看出只要其關(guān)聯(lián)的 目標(biāo)對(duì)象 不為null且不是枚舉類(lèi)型,則執(zhí)行其finalize()方法赘方;第 109 行代碼調(diào)用了 super.clear() 方法將目標(biāo)對(duì)象引用置為null烧颖,已防止重復(fù)執(zhí)行finalize方法。

虛引用(PhantomReference) 存在的意義和使用場(chǎng)景是窄陡?

這個(gè)不常用炕淮,可參看其子類(lèi)sun.misc.Cleaner。

參考文中測(cè)試代碼泳梆?傳送門(mén)~
由于水平有限鳖悠,文中錯(cuò)誤之處難免榜掌,發(fā)現(xiàn)問(wèn)題還望指出;如果你對(duì)文中觀點(diǎn)有不同看法請(qǐng)留言告訴我乘综,謝謝憎账。

?著作權(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)離奇詭異,居然都是意外死亡萌朱,警方通過(guò)查閱死者的電腦和手機(jī)宴树,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)晶疼,“玉大人酒贬,你說(shuō)我怎么就攤上這事〈浠簦” “怎么了锭吨?”我有些...
    開(kāi)封第一講書(shū)人閱讀 152,671評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)寒匙。 經(jīng)常有香客問(wèn)我零如,道長(zhǎng),這世上最難降的妖魔是什么锄弱? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,252評(píng)論 1 279
  • 正文 為了忘掉前任考蕾,我火速辦了婚禮,結(jié)果婚禮上棵癣,老公的妹妹穿的比我還像新娘辕翰。我一直安慰自己,他們只是感情好狈谊,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,253評(píng)論 5 371
  • 文/花漫 我一把揭開(kāi)白布喜命。 她就那樣靜靜地躺著,像睡著了一般河劝。 火紅的嫁衣襯著肌膚如雪壁榕。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,031評(píng)論 1 285
  • 那天赎瞎,我揣著相機(jī)與錄音牌里,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛牡辽,可吹牛的內(nèi)容都是我干的喳篇。 我是一名探鬼主播,決...
    沈念sama閱讀 38,340評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼态辛,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼麸澜!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起奏黑,我...
    開(kāi)封第一講書(shū)人閱讀 36,973評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤炊邦,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后熟史,有當(dāng)?shù)厝嗽跇?shù)林里發(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
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望眷细。 院中可真熱鬧拦盹,春花似錦、人聲如沸溪椎。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,259評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)校读。三九已至沼侣,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間歉秫,已是汗流浹背蛾洛。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(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)容

  • Java GarbageCollection(GC) Java不能像C/C++那樣直接對(duì)內(nèi)存進(jìn)行操作(內(nèi)存分配和垃...
    獅_子歌歌閱讀 2,382評(píng)論 0 3
  • 這篇文章整理自一位外國(guó)大神的英文博客,我在保存文章的結(jié)構(gòu)下椿浓,增加了一些自己的見(jiàn)解太援,并做了一個(gè)文章的腦圖。原文鏈接為...
    wean_a23e閱讀 1,206評(píng)論 0 0
  • 引用的分類(lèi) Java 1.2以后扳碍,除了普通的引用外提岔,Java還定義了軟引用、弱引用笋敞、虛引用等概念碱蒙。 強(qiáng)引用:GC ...
    劉惜有閱讀 820評(píng)論 0 1
  • 我們知道,動(dòng)態(tài)代理(這里指JDK的動(dòng)態(tài)代理)與靜態(tài)代理的區(qū)別在于夯巷,其真實(shí)的代理類(lèi)是動(dòng)態(tài)生成的赛惩。但具體是怎么生成,生...
    TheAlchemist閱讀 2,096評(píng)論 2 10
  • Activity是Android應(yīng)用中負(fù)責(zé)與用戶(hù)交互的組件趁餐,給Android應(yīng)用提供可視化用戶(hù)界面喷兼。Activit...
    Rave_Tian閱讀 1,704評(píng)論 0 1