java finalize方法總結、GC執行finalize的過程
注:本文的目的並不是鼓勵使用finalize方法,而是大致理清其作用、問題以及GC執行finalize的過程。
1. finalize的作用
- finalize()是Object的protected方法,子類可以覆蓋該方法以實現資源清理工作,GC在回收物件之前呼叫該方法。
- finalize()與C++中的解構函式不是對應的。C++中的解構函式呼叫的時機是確定的(物件離開作用域或delete掉),但Java中的finalize的呼叫具有不確定性
- 不建議用finalize方法完成“非記憶體資源”的清理工作,但建議用於:① 清理本地物件(通過JNI建立的物件);② 作為確保某些非記憶體資源(如Socket、檔案等)釋放的一個補充:在finalize方法中顯式呼叫其他資源釋放方法。其原因可見下文[finalize的問題]
2. finalize的問題
- 一些與finalize相關的方法,由於一些致命的缺陷,已經被廢棄了,如System.runFinalizersOnExit()方法、Runtime.runFinalizersOnExit()方法
- System.gc()與System.runFinalization()方法增加了finalize方法執行的機會,但不可盲目依賴它們
- Java語言規範並不保證finalize方法會被及時地執行、而且根本不會保證它們會被執行
- finalize方法可能會帶來效能問題。因為JVM通常在單獨的低優先順序執行緒中完成finalize的執行
- 物件再生問題:finalize方法中,可將待回收物件賦值給GC Roots可達的物件引用,從而達到物件再生的目的
- finalize方法至多由GC執行一次(使用者當然可以手動呼叫物件的finalize方法,但並不影響GC對finalize的行為)
3. finalize的執行過程(生命週期)
(1) 首先,大致描述一下finalize流程:當物件變成(GC Roots)不可達時,GC會判斷該物件是否覆蓋了finalize方法,若未覆蓋,則直接將其回收。否則,若物件未執行過finalize方法,將其放入F-Queue佇列,由一低優先順序執行緒執行該佇列中物件的finalize方法。執行finalize方法完畢後,GC會再次判斷該物件是否可達,若不可達,則進行回收,否則,物件“復活”。
(2) 具體的finalize流程:
物件可由兩種狀態,涉及到兩類狀態空間,一是終結狀態空間 F = {unfinalized, finalizable, finalized};二是可達狀態空間 R = {reachable, finalizer-reachable, unreachable}。各狀態含義如下:
- unfinalized: 新建物件會先進入此狀態,GC並未準備執行其finalize方法,因為該物件是可達的
- finalizable: 表示GC可對該物件執行finalize方法,GC已檢測到該物件不可達。正如前面所述,GC通過F-Queue佇列和一專用執行緒完成finalize的執行
- finalized: 表示GC已經對該物件執行過finalize方法
- reachable: 表示GC Roots引用可達
- finalizer-reachable(f-reachable):表示不是reachable,但可通過某個finalizable物件可達
- unreachable:物件不可通過上面兩種途徑可達
狀態變遷圖:
變遷說明:
- 新建物件首先處於[reachable, unfinalized]狀態(A)
- 隨著程式的執行,一些引用關係會消失,導致狀態變遷,從reachable狀態變遷到f-reachable(B, C, D)或unreachable(E, F)狀態
- 若JVM檢測到處於unfinalized狀態的物件變成f-reachable或unreachable,JVM會將其標記為finalizable狀態(G,H)。若物件原處於[unreachable, unfinalized]狀態,則同時將其標記為f-reachable(H)。
- 在某個時刻,JVM取出某個finalizable物件,將其標記為finalized並在某個執行緒中執行其finalize方法。由於是在活動執行緒中引用了該物件,該物件將變遷到(reachable, finalized)狀態(K或J)。該動作將影響某些其他物件從f-reachable狀態重新回到reachable狀態(L, M, N)
- 處於finalizable狀態的物件不能同時是unreahable的,由第4點可知,將物件finalizable物件標記為finalized時會由某個執行緒執行該物件的finalize方法,致使其變成reachable。這也是圖中只有八個狀態點的原因
- 程式設計師手動呼叫finalize方法並不會影響到上述內部標記的變化,因此JVM只會至多呼叫finalize一次,即使該物件“復活”也是如此。程式設計師手動呼叫多少次不影響JVM的行為
- 若JVM檢測到finalized狀態的物件變成unreachable,回收其記憶體(I)
- 若物件並未覆蓋finalize方法,JVM會進行優化,直接回收物件(O)
- 注:System.runFinalizersOnExit()等方法可以使物件即使處於reachable狀態,JVM仍對其執行finalize方法
4. 一些程式碼示例
(1) 物件復活
- public class GC {
- public static GC SAVE_HOOK = null;
- public static void main(String[] args) throws InterruptedException {
- SAVE_HOOK = new GC();
- SAVE_HOOK = null;
- System.gc();
- Thread.sleep(500);
- if (null != SAVE_HOOK) { //此時物件應該處於(reachable, finalized)狀態
- System.out.println("Yes , I am still alive");
- } else {
- System.out.println("No , I am dead");
- }
- SAVE_HOOK = null;
- System.gc();
- Thread.sleep(500);
- if (null != SAVE_HOOK) {
- System.out.println("Yes , I am still alive");
- } else {
- System.out.println("No , I am dead");
- }
- }
- @Override
- protected void finalize() throws Throwable {
- super.finalize();
- System.out.println("execute method finalize()");
- SAVE_HOOK = this;
- }
- }
(2)覆蓋finalize方法以確保資源釋放
作為一個補充操作,以防使用者忘記“關閉“資源,JDK中FileInputStream、FileOutputStream、Connection類均用了此”技術“,下面程式碼摘自FileInputStream類
- /**
- * Ensures that the <code>close</code> method of this file input stream is
- * called when there are no more references to it.
- *
- * @exception IOException if an I/O error occurs.
- * @see java.io.FileInputStream#close()
- */
- protected void finalize() throws IOException {
- if ((fd != null) && (fd != FileDescriptor.in)) {
- /* if fd is shared, the references in FileDescriptor
- * will ensure that finalizer is only called when
- * safe to do so. All references using the fd have
- * become unreachable. We can call close()
- */
- close();
- }
- }
參考:
12.6 Finalization of Class Instances https://notendur.hi.is//~snorri/SDK-docs/lang/lang083.htmThe Finalizable Object http://www.artima.com/interfacedesign/Finalizable.html何時及如何使用finalize
從以上的分析得出,以下結論。
1 最重要的,儘量不要用finalize,太複雜了,還是讓系統照管比較好。可以定義其它的方法來釋放非記憶體資源。
2 如果用,儘量簡單。
3 如果用,避免物件再生,這個是自己給自己找麻煩。
4 可以用來保護非記憶體資源被釋放。即使我們定義了其它的方法來釋放非記憶體資源,但是其它人未必會呼叫該方法來釋放。在finalize裡面可以檢查一下,如果沒有釋放就釋放好了,晚釋放總比不釋放好。
5 即使物件的finalize已經執行了,不能保證該物件被銷燬。要實現一些保證物件徹底被銷燬時的動作,只能依賴於java.lang.ref裡面的類和GC互動了。
相關文章
- GC和解構函式(Finalize 方法)GC函式
- java finalize方法的使用Java
- finalize方法
- 深入理解ReferenceQueue GC finalize ReferenceGC
- JEP 421: Java將要終結finalize()了!Java
- 弄清Java虛擬機器GC的執行過程Java虛擬機GC
- Java中final、finally、finalize的區別Java
- Java中final,finally,finalize的區別Java
- Java之final、finalize、finally的區別Java
- Java中final、finally和finalize的區別Java
- Java中final,finalize和finally的區別Java
- 【JAVA】筆記(6)--- toString方法;equals方法;finalize方法;package與import;內部類;Java筆記PackageImport
- final、finally、finalize的理解
- 好程式設計師Java學習路線分享finalize()方法詳解程式設計師Java
- 【Java面試題系列】:Java中final finally finalize的區別Java面試題
- final,finalize,finally 的區別
- final,finalize,finally的區別
- final、finally、finalize的區別
- Java 程式執行過程Java
- C# Dispose 和 Finalize 要點C#
- final、finally、finalize的詳細分析
- 【Java技術專題】「原理專題」深入分析Java中finalize方法的作用和底層原理Java
- 淺析Java程式的執行過程Java
- Java執行緒總結Java執行緒
- 闡述final、finally、finalize的區別
- 『Java 語法基礎』final、finalize 和 finally 的不同之處Java
- 淺談C#記憶體回收與Dispose﹐Close﹐Finalize方法[轉]C#記憶體
- Java執行緒池的增長過程Java執行緒
- Java 執行緒池的建立過程分析Java執行緒
- java多執行緒總結Java執行緒
- final、finally、finalize()的區別(skycto JEEditor)
- sql語句執行過程小結SQL
- 判斷某過程是否在執行的方法
- Java 執行過程中的記憶體模型Java記憶體模型
- 指令的執行過程
- 1.淺談final,finally,finalize的區別。
- final、finally與finalize三者的區別
- Java 多執行緒 - 總結概述Java執行緒