Java內存回收機制--Java引用的種類(強引用、弱引用、軟引用、虛引用)

Java內存管理包括內存分配(創建Java對象)和內存回收(回收Java對象)。這兩者都是JVM(Java虛擬機)自動完成,正因如此,可能很多程序員不再關心程序的內存分配。而作為一把雙刃劍,這種自動處理的機制一方面減少了Java程序開發的難度,另一方面也加重了JVM的工作負擔,使得Java程序運行相對較慢。如果程序員不注意代碼的優化,不注重減輕JVM內存分配的負擔,很容易降低程序性能,甚至會導致程序因為內存泄露而停止運行。

一、對象在內存中的狀態

JVM是否回收一個對象的標準:是否還有引用變量引用該對象?只要有引用變量引用該對象,垃圾回收機制就不會回收它。

JVM的內存回收機制采用有向圖方式來管理內存中的對象,對于單線程程序而言,整個程序基于main線程,那么該有向圖就是以main進程為頂點的有向圖。在此有向圖中,main頂點可達的對象處于可達狀態,垃圾回收機制不會回收它們,如果在有向圖中,main線程不可達,則垃圾回收機制就會主動回收它了。

如下程序:

classNode?{

Node?next;

String?name;

publicNode(String?name)?{

this.name?=?name;

}

}

publicclassNodeTest?{

publicstaticvoidmain(String[]?args)?{

Node?n1?=newNode("第一個節點");

Node?n2?=newNode("第二個節點");

Node?n3?=newNode("第三個節點");

n1.next?=?n2;

n3?=?n2;

n2?=null;

}

}

對應的有向圖如下:

則很明顯,“第三個Node對象”不可達,接下來就會成為垃圾回收機制的回收對象。

另外值得一提的是,有向圖的管理方式能有效解決循環引用的問題,例如,有三個對象相互引用,A引用B,B引用C,C引用A,他們都沒有失去引用,但是只要從有向圖的頂點(也就是進程根)不可達,則垃圾回收機制就會回收它們。采用有向圖管理內存對象,有較高精度,但缺點是效率較低。

根據對象在有向圖中的狀態,可以分為三種:

可達狀態:對象被創建后有一個以上的引用變量引用它。程序可通過該引用變量來調用該對象的屬性和方法。

可恢復狀態:程序中某個對象不再有任何引用變量引用,它將進入可恢復狀態,此時有向圖的起始頂點不能導航到該對象。在該狀態下,系統的垃圾回收機制準備回收該對象所占用 的內存。在回收該對象之前,系統會調用可恢復狀態的對象的finalize方法進行資源清理,如果系統在調用finalize方法重新讓一個以上引用變量引用該對象,則這個對象會再次變為可達狀態;否則,該對象進入不可達狀態。

不可達狀態:對象 的所有關聯都被切斷,且系統調用對象的finalize方法依然沒有使對象變為可達狀態,那這個對象將永久性地失去引用,變成不可達狀態。只有當一個對象處于不可達狀態時,系統才會真正回收該對象所占有的資源。

三種狀態的轉換圖如下所示:

值得注意的是,一個對象可以被一個方法局部變量所引用,也可以被其他類的類變量引用,或者被其他對象的實例變量所引用。當某個對象被其他類的類變量所引用時,只有該類被銷毀后,該對象才會進入可恢復狀態。當某個對象被其他對象的實例變量引用時,只有當引用該對象的對象被銷毀或進入不可達狀態時,該對象才會進入不可達狀態。

二、幾種引用方式:

在JDK1.2版本開始,把對象的引用分為四種級別,從而使程序更加靈活地控制對象的生命周期,這四種級別由高到低依次為:強引用、軟引用、弱引用、虛引用。相關對應的類圖結構如下:

1、強引用

以前我們使用的大部分引用實際上都是強引用,這是使用最普遍的引用。如果一個對象具有強引用,就表示它處于可達狀態,垃圾回收器絕不會回收它,即便系統內存非常緊張,Java虛擬機寧愿拋出OutOfMemoryError錯誤,使程序異常終止,也不會回收被強引用所引用的對象。因此,強引用是造成Java內存泄露的主要原因之一。

2、軟引用(SoftReference)

當一個對象只具有軟引用,如果內存空間足夠,垃圾回收器就不會回收它,如果內存空間不足了,就會回收這些對象的內存。只要垃圾回收器沒有回收它,該對象就可以被程序使用。軟引用可用來實現內存敏感的高速緩存。

軟引用可以和一個引用隊列(ReferenceQueue)聯合使用,如果軟引用所引用的對象被垃圾回收,JAVA虛擬機就會把這個軟引用加入到與之關聯的引用隊列中。

3.弱引用(WeakReference)

如果一個對象只具有弱引用,那就類似于是可有可無的。弱引用和軟引用很像,但弱引用的引用級別更低。弱引用與軟引用的區別在于:只具有弱引用的對象擁有更短暫的生命周期。在垃圾回收器線程掃描它所管轄的內存區域的過程中,一旦發現了只具有弱引用的對象,不管當前內存空間足夠與否,都會回收它的內存。不過,由于垃圾回收器是一個優先級較低的線程, 因此不一定會很快發現那些只具有弱引用的對象,即只有等到系統垃圾回收機制運行時才會被回收。

弱引用可以和一個引用隊列(ReferenceQueue)聯合使用,如果弱引用所引用的對象被垃圾回收,Java虛擬機就會把這個弱引用加入到與之關聯的引用隊列中。

4.虛引用(PhantomReference)

"虛引用"顧名思義,就是形同虛設,與其他幾種引用都不同,虛引用并不會決定對象的生命周期。如果一個對象僅持有虛引用,那么它就和沒有任何引用一樣,在任何時候都可能被垃圾回收,程序也并不能通過虛引用訪問被引用的對象。

虛引用主要用來跟蹤對象被垃圾回收的狀態。虛引用與軟引用和弱引用的一個區別在于:虛引用不能單獨使用,虛引用必須和引用隊列(ReferenceQueue)聯合使用。當垃圾回收器準備回收一個對象時,如果發現它還有虛引用,就會在回收對象的內存之前,把這個虛引用加入到與之關聯的引用隊列中。程序可以通過判斷引用隊列中是否已經加入了虛引用,來了解被引用的對象是否將要被垃圾回收。程序如果發現某個虛引用已經被加入到引用隊列,那么就可以在所引用的對象的內存被回收之前采取必要的行動。

各類引用說明對比如下表所示:

使用這些引用類可以避免程序執行期間將對象留在內存中。如果以軟引用、弱引用或虛引用的方式引用對象,垃圾回收器就能隨意地釋放對象。如果希望盡可能減小程序在其生命周期中所占用的內存大小,這些引用類就很有好處。

轉載自:http://blog.csdn.NET/daijin888888/article/details/49949283

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,406評論 6 538
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,034評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 177,413評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,449評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,165評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,559評論 1 325
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,606評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,781評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,327評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,084評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,278評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,849評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,495評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,927評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,172評論 1 291
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,010評論 3 396
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,241評論 2 375

推薦閱讀更多精彩內容

  • 在Java中,它的內存管理包括兩方面:內存分配(創建Java對象的時候)和內存回收,這兩方面工作都是由JVM自動完...
    juexin閱讀 293評論 0 1
  • Java內存回收機制 ?1.對象的創建與可達 ??Java中,所有對象都是通過new關鍵字來創建的,JVM會在堆內...
    我是java程序員閱讀 869評論 0 1
  • 參考網址:http://www.blogjava.net/ldwblog/archive/2013/07/24/4...
    felix_feng閱讀 773評論 0 2
  • 一、垃圾回收機制的意義Java語言中一個顯著的特點就是引入了垃圾回收機制,使c++程序員最頭疼的內存管理的問題迎刃...
    任任任任師艷閱讀 653評論 0 0
  • 簡介 Java內存回收機制 不論哪種語言的內存分配方式,都需要返回所分配內存的真實地址,也就是返回一個指針到內存塊...
    梅_梅閱讀 239評論 0 0