Java虛擬機(jī)中有著垃圾回收機(jī)制,自動幫我們回收對象,不需要自己手動實(shí)現(xiàn)垃圾回收。那么JVM中的垃圾收集器是如何判斷對象是否需要回收的呢?這將是本篇文章的主要內(nèi)容。
一、引用計(jì)數(shù)法
給對象添加一個引用計(jì)數(shù)器,每當(dāng)有一個地方引用它時,計(jì)數(shù)器值就加1;當(dāng)引用失效時,計(jì)數(shù)器值就減1;任何時刻計(jì)數(shù)器為0的對象就是不可能被再使用的。
主流的JVM里面沒有選用引用計(jì)數(shù)算法來管理內(nèi)存,其中最主要的原因是它很難解決對象間的互循環(huán)引用的問題。
二、可達(dá)性分析算法
通過一些列的稱為“GC Roots”的對象作為起始點(diǎn),從這些節(jié)點(diǎn)開始向下搜索,搜索所走過的路徑稱為引用鏈,當(dāng)一個對象到GC Roots沒有任何引用鏈相連時(就是從GC Roots 到這個對象是不可達(dá)),則證明此對象是不可用的。所以它們會被判定為可回收對象(例如圖B中的對象既是不可達(dá)的)。
在可達(dá)性分析算法中,要真正宣告一個對象死亡,至少要經(jīng)歷兩次標(biāo)記過程:
1.如果對象在進(jìn)行可達(dá)性分析后發(fā)現(xiàn)沒有與GC Roots相連接的引用鏈,那它將會被第一次標(biāo)記并且進(jìn)行一次篩選,篩選的條件是此對象是否有必要執(zhí)行finalize()方法。當(dāng)對象沒有 覆蓋finalize()方法,或者finalize()方法已經(jīng)被虛擬機(jī)調(diào)用過,虛擬機(jī)將這兩種情況都視為“沒有必要執(zhí)行”。
2.如果這個對象被判定為有必要執(zhí)行finalize()方法,那么這個對象將會放置在一個叫做F-Queue隊(duì)列之中,并在稍后由一個由虛擬機(jī)自動建立的、低優(yōu)先級的Finalizer線程去執(zhí)行它。finalize()方法是對象逃脫死亡命運(yùn)的最后一次機(jī)會,稍候GC將對F-Queue中的對象進(jìn)行第二次小規(guī)模的標(biāo)記,如果對象要在finalie()中成功拯救自己——只要重新與引用鏈上的任何一個對象建立關(guān)聯(lián)即可,譬如把自己(this關(guān)鍵字)賦值給某個類變量或者對象的成員變量,那在第二次標(biāo)記時它將會被移除出“即將回收”的集合;如果對象這時候還沒有逃脫,那基本上它就真的被回收了。
三、判斷對象是否存活與“引用”有關(guān)
在JDK1.2之后,Java對引用的概念進(jìn)行了擴(kuò)充,將引用分為強(qiáng)引用(Strong Reference)、軟引用(Soft Reference)、弱引用(Weak Reference)、虛引用(Phantom Reference)四種,這四種引用強(qiáng)度依次逐漸減弱。
強(qiáng)引用:就是指在程序代碼之中普遍存在的,類似“Object obj = new Object()”這類的引用,只要強(qiáng)引用還存在,垃圾收集器永遠(yuǎn)不會回收掉被引用的對象。
軟引用:用來描述一些還有用但并非必須的對象。在系統(tǒng)將要發(fā)生內(nèi)存溢出異常之前,將會把這些對象列進(jìn)回收范圍之中進(jìn)行第二次回收。
弱引用:用戶描述非必須對象的。被弱引用關(guān)聯(lián)的對象只能生存到下一次垃圾收集發(fā)生之前。當(dāng)垃圾收集器工作時,無論當(dāng)前內(nèi)存是否足夠,都會回收掉只被弱引用關(guān)聯(lián)的對象。
虛引用:一個對象是否有虛引用存在,完全不會對其生存時間構(gòu)成影響,也無法通過虛引用來取得一個對象實(shí)例。為一個對象設(shè)置虛引用的唯一目的就是能在這個對象被收集器回收時刻得到一個系統(tǒng)通知。