天天看點

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

原文連結

對象的誕生

“你醒啦!”,迷迷糊糊中聽到一個聲音,我睜開了眼睛,發現一個小夥伴正看着我。

“這裡是哪裡,你是誰啊?”“這裡是堆區,我是一個Ajax對象,叫我小A吧”

我慢慢坐了起來,舉目四望,這裡有好多形形色色的對象居住在這裡,遠處還有好多的線程在各自忙碌着,好一副熱鬧的景象!

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

“你好,我才剛醒,我還不知道我是什麼對象呢”“這個簡單,讓我看看哈~~哦,原來你是一個APIController對象啊”,小A摸了摸我的頭。“你怎麼知道的?”

“你的頭上這裡有個64bit的Klass指針,喏,順着這個方向看過去,那裡記錄了你所屬的類資訊,你看,那裡寫着APIController呢”

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

聽他這麼一說,我這才注意到我的頭上有兩個64bit的字段。

“唉,小A,旁邊這個64bit的數字又是裝的什麼内容呢?”“那個叫MarkWord,是咱們Java對象的門面,裡面的資訊可重要了,你可要保管好了,這裡面有。。。”

Minor GC

突然,不知從哪裡傳來一串警報聲,随後聽到廣播:“各線程注意,請進入安全點等待,各分區注意,啟動一次Minor GC”

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

聽到廣播的我莫名的緊張起來。

不知怎麼回事,遠處忙碌的線程們都仿佛被施了定身法一般,都停下了手頭的活。隻有少數幾個還在活動,這幾個線程大叔看上去跟他們有些不一樣,其中有幾個朝我們這邊走了過來。

“這是要幹什麼啊?”我向小A打聽情況。“我也不知道,我也比你先出生沒多久,這情況我也是第一次見到”,小A好像也有一點慌張。

沒過多久,來了一個兇巴巴的管理者線程,拿着喇叭吼着:“Eden區的對象們聽着,念到名字的站起來”說完,便開始一個個點名,心裡一陣忐忑,怕被叫到,又怕不被叫到。

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

念了很久,終于聽到了小A和我的名字,我倆戰戰兢兢的站了起來。

沒多久就念完了,我一瞅,站起來的是少數啊,心裡有點不好的預感。“念到名字的跟我來,其他的交給我的助手處理”,說完大家跟着他開始移動。

在走的路上,碰上了另外一支隊伍,和我們彙合了。

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

“唉,兄弟,怎麼稱呼,你們哪個機關的?”小A熱情的上去和一個對象攀談了起來。“叫我小B吧,我們這波是Survivor From區的,你們Eden區來的吧,我半小時前還在你們那兒待過呢”,這個自稱是小B的也很随和。

“小B哥您好,咱們這是要去哪裡啊?”我也上前搭了句話。“前面是Survivor To區”

“咱們是怎麼被挑出來的?”“這裡的管理者會通過一種叫GC Roots的對象,順藤摸瓜,找出所有還有引用關系的對象,咱們就是幸存者,說明咱們還有價值”

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

“那留下的對象怎麼辦?”“他們的命運多半懸了,因為沒有别的對象引用他們了,需要把他們清理掉,騰出空間來”

我似懂非懂,一邊走一邊擔心着,很快我們就到了傳說中的Survivor To區,管理者安排我們都坐下,“這裡好小啊”

“那可不,比起你原來的Eden區,這裡隻有八分之一大小”,我一回頭,剛才路上碰到的小B居然就在我和小A的旁邊,巧了不是。“唉,小B哥,咱們這麼折騰一圈是在做什麼啊?”

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

“這叫做垃圾回收GC,你們開始待那地方叫Eden區,對象出生的地方都在那裡。咱們所在的地方是一個叫Java Virtual Machine的世界,程式員隻管建立對象,不管釋放,這對象越來越多,Eden區放不下了,自然就要騰出空間來了。”

我和小A都點了點頭,心裡慶幸躲過一劫,擡頭望去,不知什麼時候,那些定住的線程們又開始忙活起來了。

“還沒恭喜你們呢,長大一歲了”,小B拍了拍我倆的肩膀,我倆面面相觑,滿臉問号。“這是從何說起呢?”,小A先開口了。

“你們頭上的MarkWord第3-6位記錄的就是你們的年齡,經過一次GC就長大一歲了!”我倆互相看了看,又看了看小B的GC年齡位置,居然已經15歲了。

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

“小B哥,難怪你見多識廣,都一把年紀了呀。咦,這表示GC年紀的隻有4位,最大隻能表示到15,等會兒要是再來一次GC,這不要裝不下了嗎?”,看着小B的腦袋,我陷入了思考。“再來一次GC我要是還能幸存,我就要進入老年代區域了,就不能陪你們玩兒了”,小B看着我們眨了眨眼睛。

“老年代,那是什麼地方,我們不能去嗎?”“都說了是老年代了,是我這種老年對象去的地方,你們新來的還要在Survivor To區和Survivor From區兜兜轉轉好些回合呢,等你們到我這把年紀就能過去了”

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

“啊,為什麼這麼麻煩,設定這麼多區都是幹嘛的啊?”小A急着問。小B把手搭在小A的肩說到:“這裡的管理者用的是标記-複制算法來清理空間,是以需要在Eden區之外再設一個地方接收複制活下來的對象。”

“那加一個Survivor區就夠了啊,幹嘛弄兩個Survivor區?”我也抛了一個問題。小B把另一隻手搭在了我的肩上,“這是為了讓存活的對象能夠在這邊反複流轉,不要着急去老年代區域”

“那為什麼Survivor比Eden小那麼多?”,我繼續問到“根據他們統計發現,98%的對象都活不過一輪GC,留下來的都是少數。而且兩個Survivor區有一個要空着,如果太大就太浪費了。”

聽着小B的話我們倆都陷入了沉思。

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

沒過一會兒,廣播又響了起來:“各線程注意,請進入安全點等待,各分區注意,啟動一次Minor GC”,剛剛平靜的心又一次懸了起來。管理者又開始點名,這一次,我和小B都被點到了,而沒有聽到小A的名字。

我們跟小A告别了,離開了Survivor To區,走到分叉路口,小B也跟我道别:“再見了朋友,如果有機會,老年代等你來再聚”

接下來就隻剩我一個對象了,跟随陌生的對象隊伍來到了Survivor From區,這裡跟剛才的To區規模相當,隻是隊伍比起之前那次又小了很多。

來到自己的位置坐好,看了看頭上的GC年齡位,我2歲了。

Finalizer對象

沒有了熟悉的朋友,獨自發着呆,等待着線程們來通路我。突然,有人拍了拍我的肩膀,我回頭看去,居然是小A,他跑的氣喘籲籲的。

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

“你不是沒有被念到名字,沒有對象再引用你了嗎,居然沒有被清理?”,再次看見小A,我有點難以置信。“剛才真的好險,我都吓死了,沒想到事情出現了轉機!”

“發生了什麼,快告訴我!”,我迫不及待的想知道這一切究竟是怎麼回事。小A喘了幾口氣繼續說到:“就在你們走後,管理者又拿出了另外一份名單,我的名字居然在上面,我一打聽才知道原來檢查到有一個Finalizer對象還在引用我~”

“奇怪了,不是說沒有對象嗎,怎麼又冒出了一個Finalizer對象?這是什麼?”“後來我見到了那個Finalizer對象,就在開始我們沒多遠的位置,聽他說是因為我所屬的類有覆寫finalize方法,是以在我出生的時候,他也一同誕生,并且一直持有我的引用”

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

“那後來呢?”,我繼續追問。“後來啊,他被管理者放進了一個ReferenceQueue的隊列去了,他們把那地方叫F-Queue監獄。等待一個名字也叫Finalizer的線程大叔去處理,通過Finalizer對象來調用我的finalize方法,之後就把我們之間的聯系切斷了”“他要倒黴了!不過好在還是救了你一命”“躲得了初一,躲不過十五,現在連Finalizer對象也沒引用我了,下一次GC我鐵定要完蛋的”,小A說完又低下了頭。“别想那麼多,做對象最重要的就是開心了,說不定下一次我陪你一起完蛋呢”

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

我倆剛剛說完,熟悉的廣播又想起了:“各線程注意,請進入安全點等待,各分區注意,啟動一次Minor GC”很快,管理者就念到了我的名字,看來我還能撐下去。快到結束的時候,管理者居然神奇般的念到了小A的名字。

“小A,你聽到了嗎,居然還有對象在引用你!”我高興的對小A說到,小A也使勁點點頭。“我知道了,一定是Finalizer線程大哥在執行我的finalize方法的時候,又把我和誰建立了聯系,對,一定是這樣!”

我們再一次從To區來到了From區,這一次又少了很多舊面孔,不過從Eden區來了不少新面孔。

往後的一段時間裡,我們在這兜兜轉轉了好多輪,終于看到頭上的年齡标記慢慢變成了15歲。

Full GC

沒過多久,廣播再次響起,我和小A幸運的再次被點到名字,随後,管理者檢查了我和小A,發現我倆超齡了,直接給我們趕到了一條新的路上,我知道前面就是傳說中的老年代了。

來到這個陌生的地方,放眼望去,這裡可比我待過的Eden區、From區、To區加起來還要大,裡面有好多的對象,密密麻麻的,不過看上去一個個都不是省油的燈,畢竟能來到這裡的對象,比起Eden區的那些萌新都是些老油條了。

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

我倆找到自己的位置坐下開始閑聊,這時,從不遠處走過來一個身影,走近一看正是之前的小B。“唉,這不是小A嗎,記得你不是被清理了嗎,怎麼來這兒了?”,對于小A的出現,小B哥有些驚訝。

随後小A也向小B聊起了之前的那段驚險經曆~正在閑聊的時候,管理者突然進來圈了一大片空閑的位置,建立了一個巨大的對象!衆對象都驚呆了!

【趣話程式設計】一個Java對象的回憶錄:垃圾回收

“小B哥,這哥們什麼來路,空降啊,直接到老年代!”我好奇的問到。“沒辦法,誰叫人家體格大,Eden區要麼裝不下,要麼嫌給他複制過去複制過來太費勁,是以直接給安排到這裡了,不像我們要一點點熬過來。”

“各線程注意,請進入安全點等待,各分區注意,啟動一次Full GC“,熟悉的廣播又一次響起,隻不過這一次聽到的是Full GC。“看來記憶體吃緊了啊!”小B歎了口氣。

管理者又開始點名了,這一次,幸運不再眷顧小A。

未完待續~~

彩蛋

一個線程小姐姐迎面向我走了過來。

“Hi,小朋友你好,我是3002号線程,現在我要來鎖定你,别動哦,讓我檢查下你的MarkWord”“lock位是01,Good!讓我再看看偏向鎖标記位,呀!是個1,糟糕”,小姐姐皺起了眉頭。

欲知後事如何,請關注後續精彩......

來源 | 程式設計技術宇宙

作者 | 軒轅之風