標籤:rom gpo [1] weak 不清楚 它的 cloc reference ref
之前在看深入理解Java虛擬機器一書中第一次接觸相關名詞,但是並不理解,只知道Object obj = new Object()類似這種操作的時候,obj就是強引用。強引用不會被gc回收直到gc roots不可達時。而對其他三個名詞並不清楚,因為並沒有被真正使用過。通過查看軟引用,弱引用和虛引用的源碼,可以看出這三個類都是繼承自Reference。
一 概念
1.1軟引用(SoftReference)
我理解的軟引用的意思是,即使引用對象沒有被使用了,gc也不會馬上回收,而是只有當堆記憶體空間不夠時才會回收。一般用於緩衝對象,因為希望在記憶體中保留的越久越好。軟引用所指向的引用對象停留在堆記憶體的時間由當前可用堆記憶體的大小來控制。軟引用中有兩個欄位:clock和timestamp。clock記錄初始化或gc的時間,而get引用對象時會更新timestamp與clock時間一致。如果clock和timestamp的間隔時間超過maxInternal,則認為該引用對象很久沒有使用是時候回收該引用對象,否則會繼續存在。maxInternal的時間有當前可用堆記憶體大小決定。具體jvm源碼可以參考:https://mp.weixin.qq.com/s/pPXNeUI57GD-YnnfQWGXbg
1.2弱引用 (WeakReference)
弱引用的意思是如果引用對象gc roots不可達,gc就會進行回收。get操作可能返回引用對象(如果沒有gc),否則返回空。
1.3虛引用 (PhantomReference)
虛引用不能返回引用對象,因為get操作返回的一直是null,感覺就是一個虛無縹緲的引用,所以這是稱之為虛引用的原因吧。
二 執行個體
構造一個對象TestObj,它的大小為1M對一點。
JVM參數設定:
-Xms8M 初始堆大小
-Xmx8M 堆的最大值
-Xmn4M 年輕代的大小
-XX:+PrintGCDetails
-XX:SurvivorRatio=8 eden:survivor的比列為8:1
第一種情況:
gc過程:
分析:
Eden:3072k, from:512k, to:512k, ParOldGen:4096k
調用System.gc的時候對象強引用還存在,所以三個引用對象會進入到老年代。
第二種情況:
gc過程:
分析:
由於此時年輕代不能再分配一個1M多的空間,而老年代也放不下,所以會發生full gc, 弱引用和虛引用的對象會被回收,而軟引用還存在。cost[0],cost[1]進入老年代,而此時年輕代應該只有一個cost[2]對象,為什麼會佔據eden:3072k空間的69%呢,這讓人很疑惑???
第三種情況:
gc過程:
分析:
由於eden:3072k的空間已經被使用了69%,所以不能在放下一個1M的對象,只能full gc(當準備要觸發一次young GC時,如果發現統計資料說之前young GC的平均晉陞大小比目前old gen剩餘的空間大,則不會觸發young GC而是轉為觸發full GC。full gc 和 young gc關係參考:https://www.zhihu.com/question/41922036/answer/93079526),所以軟引用所指對象被最終收回。果然軟引用存活時間夠長,只有堆記憶體足夠。
java中的強引用(Strong reference),軟引用(SoftReference),弱引用(WeakReference),虛引用(PhantomReference)