java ----------finalize方法總結、GC執行finalize的過程

來源:互聯網
上載者:User

標籤:href   nali   cli   流程   消失   view   details   roo   connect   

java finalize方法總結、GC執行finalize的過程分類: Java2013-10-06 16:42 73人閱讀 評論(0) 收藏 舉報finalizejavajvm 

目錄(?)[+]

 

 

註:本文的目的並不是鼓勵使用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:對象不可通過上面兩種途徑可達
狀態變遷圖:變遷說明:
  1. 建立對象首先處於[reachable, unfinalized]狀態(A)
  2. 隨著程式的運行,一些參考關聯性會消失,導致狀態變遷,從reachable狀態變遷到f-reachable(B, C, D)或unreachable(E, F)狀態
  3. 若JVM檢測到處於unfinalized狀態的對象變成f-reachable或unreachable,JVM會將其標記為finalizable狀態(G,H)。若對象原處於[unreachable, unfinalized]狀態,則同時將其標記為f-reachable(H)。
  4. 在某個時刻,JVM取出某個finalizable對象,將其標記為finalized並在某個線程中執行其finalize方法。由於是在活動線程中引用了該對象,該對象將變遷到(reachable, finalized)狀態(K或J)。該動作將影響某些其他對象從f-reachable狀態重新回到reachable狀態(L, M, N)
  5. 處於finalizable狀態的對象不能同時是unreahable的,由第4點可知,將對象finalizable對象標記為finalized時會由某個線程執行該對象的finalize方法,致使其變成reachable。這也是圖中只有八個狀態點的原因
  6. 程式員手動調用finalize方法並不會影響到上述內部標記的變化,因此JVM只會至多調用finalize一次,即使該對象“複活”也是如此。程式員手動調用多少次不影響JVM的行為
  7. 若JVM檢測到finalized狀態的對象變成unreachable,回收其記憶體(I)
  8. 若對象並未覆蓋finalize方法,JVM會進行最佳化,直接回收對象(O)
  9. 註:System.runFinalizersOnExit()等方法可以使對象即使處於reachable狀態,JVM仍對其執行finalize方法
4. 一些程式碼範例
(1) 對象複活 [java] view plaincopy  
  1. public class GC {  
  2.   
  3.     public static GC SAVE_HOOK = null;  
  4.   
  5.     public static void main(String[] args) throws InterruptedException {  
  6.         SAVE_HOOK = new GC();  
  7.         SAVE_HOOK = null;  
  8.         System.gc();  
  9.         Thread.sleep(500);  
  10.         if (null != SAVE_HOOK) { //此時對象應該處於(reachable, finalized)狀態  
  11.             System.out.println("Yes , I am still alive");  
  12.         } else {  
  13.             System.out.println("No , I am dead");  
  14.         }  
  15.         SAVE_HOOK = null;  
  16.         System.gc();  
  17.         Thread.sleep(500);  
  18.         if (null != SAVE_HOOK) {  
  19.             System.out.println("Yes , I am still alive");  
  20.         } else {  
  21.             System.out.println("No , I am dead");  
  22.         }  
  23.     }  
  24.   
  25.     @Override  
  26.     protected void finalize() throws Throwable {  
  27.         super.finalize();  
  28.         System.out.println("execute method finalize()");  
  29.         SAVE_HOOK = this;  
  30.     }  
  31. }  
(2)覆蓋finalize方法以確保資源釋放作為一個補充操作,以防使用者忘記“關閉“資源,JDK中FileInputStream、FileOutputStream、Connection類均用了此”技術“,下面代碼摘自FileInputStream類 [java] view plaincopy  
  1. /** 
  2.  * Ensures that the <code>close</code> method of this file input stream is 
  3.  * called when there are no more references to it. 
  4.  * 
  5.  * @exception  IOException  if an I/O error occurs. 
  6.  * @see        java.io.FileInputStream#close() 
  7.  */  
  8. protected void finalize() throws IOException {  
  9.     if ((fd != null) &&  (fd != FileDescriptor.in)) {  
  10.         /* if fd is shared, the references in FileDescriptor 
  11.          * will ensure that finalizer is only called when 
  12.          * safe to do so. All references using the fd have 
  13.          * become unreachable. We can call close() 
  14.          */  
  15.         close();  
  16.     }  
  17. }  
參考:
12.6 Finalization of Class Instances https://notendur.hi.is//~snorri/SDK-docs/lang/lang083.htm深入理解java的finalize http://zhang-xzhi-xjtu.iteye.com/blog/484934The Finalizable Object  http://www.artima.com/interfacedesign/Finalizable.html

java ----------finalize方法總結、GC執行finalize的過程

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.