欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

深入了解Java中finalize方法的作用和底層原理

 更新時間:2022年12月29日 10:58:50   作者:洛神灬殤  
這篇文章主要為大家詳細介紹了Java中finalize方法的作用和底層原理,文中的示例代碼講解詳細,具有一定的學習價值,需要的可以參考一下

finalize方法是什么

finalize方法是Object的protected方法,Object的子類們可以覆蓋該方法以實現(xiàn)資源清理工作,GC在首次回收對象之前調用該方法。

finalize方法與C++的析構函數(shù)的區(qū)別

finalize方法與C++中的析構函數(shù)不是對應的,C++中的析構函數(shù)調用的時機是確定的(對象離開作用域或delete掉),但Java中的finalize的調用具有不確定性,不建議用finalize方法完成“非內存資源”的清理工作。

finalize方法合適清理的對象

  • 清理本地對象(通過JNI創(chuàng)建的對象);
  • 作為確保某些非內存資源(如Socket、文件等)釋放的一個補充,在finalize方法中顯式調用其他資源釋放方法。

可以觸發(fā)finalize執(zhí)行的方法

在Java中含有一些一些與finalize相關的方法,由于一些致命的缺陷,已經被廢棄了,如System.runFinalizersOnExit() 方法、Runtime.runFinalizersOnExit() 方法、
System.gc() 與System.runFinalization() 方法。

他們增加了finalize方法執(zhí)行的機會,但不可盲目依賴它們Java語言規(guī)范并不保證finalize方法會被及時地執(zhí)行、而且根本不會保證它們會被執(zhí)行finalize方法可能會帶來性能問題。

因為JVM通常在單獨的低優(yōu)先級線程中完成finalize的執(zhí)行。

finalize實現(xiàn)對象再生問題

finalize方法的實現(xiàn)中,可將待回收對象賦值給GC Roots可達的對象引用,從而達到對象再生的目的。

finalize方法至多由GC執(zhí)行一次(用戶當然可以手動調用對象的finalize方法,但并不影響GC對finalize的行為)。

finalize的執(zhí)行過程(生命周期)

大致描述一下finalize的運行流程:當對象變成(GC Roots)不可達時,GC會判斷該對象是否覆蓋了finalize方法,若未覆蓋,則直接將其回收。

若對象未執(zhí)行過finalize方法,將其放入F-Queue隊列,由低優(yōu)先級線程執(zhí)行該隊列中對象的finalize方法。執(zhí)行finalize方法完畢后,GC會再次判斷該對象是否可達,若不可達,則進行回收,否則,對象“復活”。

對象對于finalize方法的兩種狀態(tài)

對象可由兩種狀態(tài),涉及到兩類狀態(tài)空間,一是終結狀態(tài)空間 F = {unfinalized, finalizable, finalized};二是可達狀態(tài)空間 R = {reachable, finalizer-reachable, unreachable}。

終結狀態(tài)空間

各狀態(tài)含義如下:

  • unfinalized: 新建對象會先進入此狀態(tài),GC并未準備執(zhí)行其finalize方法,因為該對象是可達的。
  • finalizable: 表示GC可對該對象執(zhí)行finalize方法,GC已檢測到該對象不可達。正如前面所述,GC通過F-Queue隊列和一專用線程完成finalize的執(zhí)行。

對應的流程圖如下所示:

可達狀態(tài)空間

各狀態(tài)含義如下:

  • finalized: 表示GC已經對該對象執(zhí)行過finalize方法
  • reachable: 表示GC Roots引用可達
  • finalizer-reachable(f-reachable):表示不是reachable,但可通過某個finalizable對象可達
  • unreachable:對象不可通過上面兩種途徑可達

狀態(tài)變遷圖:

變遷說明:

1.新建對象首先處于[reachable, unfinalized]狀態(tài)(A)

2.隨著程序的運行,一些引用關系會消失,導致狀態(tài)變遷,從reachable狀態(tài)變遷到f-reachable(B, C, D) 或 unreachable(E, F)狀態(tài)

3.JVM檢測到處于unfinalized狀態(tài)的對象變成f-reachable或unreachable。

JVM會將其標記為finalizable狀態(tài)(G,H)。若對象原處于[unreachable, unfinalized]狀態(tài),則同時將其標記為f-reachable(H)。

在某個時刻,JVM取出某個finalizable對象,將其標記為finalized并在某個線程中執(zhí)行其finalize方法。

4.由于是在活動線程中引用了該對象,該對象將變遷到(reachable, finalized)狀態(tài)(K或J)。該動作將影響某些其他對象從f-reachable狀態(tài)重新回到reachable狀態(tài)(L, M, N)處于finalizable狀態(tài)的對象不能同時是unreahable的。

5.將對象finalizable對象標記為finalized時會由某個線程執(zhí)行該對象的finalize方法,致使其變成reachable。

注:System.runFinalizersOnExit()等方法可以使對象即使處于reachable狀態(tài),JVM仍對其執(zhí)行finalize方法

代碼示例

對象復活

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)狀態(tài)  
            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;  
    }  
}

覆蓋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();
    }
}

注意:我們自己手動調用finalize方法并不會影響到上述內部標記的變化,因此JVM只會至多調用finalize一次,即使該對象“復活”也是如此。我們手動調用多少次不影響JVM的行為

若JVM檢測到finalized狀態(tài)的對象變成unreachable,回收其內存(I),若對象并未覆蓋finalize方法,JVM會進行優(yōu)化,直接回收對象(O)

到此這篇關于深入了解Java中finalize方法的作用和底層原理的文章就介紹到這了,更多相關Java finalize方法內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • Spring中ApplicationEventPublisher發(fā)布訂閱模式的實現(xiàn)

    Spring中ApplicationEventPublisher發(fā)布訂閱模式的實現(xiàn)

    本文主要介紹了Spring中ApplicationEventPublisher發(fā)布訂閱模式的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-07-07
  • Java實現(xiàn)幾種序列化方式總結

    Java實現(xiàn)幾種序列化方式總結

    本篇文章主要介紹了Java實現(xiàn)幾種序列化方式總結,包括Java原生以流的方法進行的序列化、Json序列化、FastJson序列化、Protobuff序列化,有興趣的可以了解一下,
    2017-03-03
  • Java實現(xiàn)調用外部程序的示例代碼

    Java實現(xiàn)調用外部程序的示例代碼

    本文主要介紹了Java實現(xiàn)調用外部程序的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-05-05
  • java打包成可執(zhí)行的jar或者exe的詳細步驟

    java打包成可執(zhí)行的jar或者exe的詳細步驟

    Java程序完成以后,對于Windows操作系統(tǒng),習慣總是想雙擊某個exe文件就可以直接運行程序,現(xiàn)我將一步一步的實現(xiàn)該過程.最終結果是:不用安裝JRE環(huán)境,不用安裝數(shù)據(jù)庫,直接雙擊一個exe文件,就可以運行程序
    2014-04-04
  • 學習SpringBoot容器功能及注解原理

    學習SpringBoot容器功能及注解原理

    這篇文章主要介紹了學習SpringBoot容器功能及注解原理,文中通過詳細的代碼示例對SpringBoot容器功能及注解原理進行了解析,有需要的朋友可以借鑒參考下
    2021-09-09
  • SpringBoot使用mybatis-plus分頁查詢無效的問題解決

    SpringBoot使用mybatis-plus分頁查詢無效的問題解決

    MyBatis-Plus提供了很多便捷的功能,包括分頁查詢,本文主要介紹了SpringBoot使用mybatis-plus分頁查詢無效的問題解決,具有一定的參考價值,感興趣的可以了解一下
    2023-12-12
  • JVM 方法調用之動態(tài)分派(詳解)

    JVM 方法調用之動態(tài)分派(詳解)

    下面小編就為大家?guī)硪黄狫VM 方法調用之動態(tài)分派(詳解)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-05-05
  • Java中單例模式的7種寫法

    Java中單例模式的7種寫法

    這篇文章主要介紹了Java中單例模式的7種寫法,本文分別給出每種方式的實現(xiàn)代碼,需要的朋友可以參考下
    2015-05-05
  • SpringBoot的WebSocket實現(xiàn)單聊群聊

    SpringBoot的WebSocket實現(xiàn)單聊群聊

    這篇文章主要為大家詳細介紹了SpringBoot的WebSocket實現(xiàn)單聊群聊,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-02-02
  • springboot 使用zookeeper實現(xiàn)分布式隊列的基本步驟

    springboot 使用zookeeper實現(xiàn)分布式隊列的基本步驟

    這篇文章主要介紹了springboot 使用zookeeper實現(xiàn)分布式隊列,通過ZooKeeper的協(xié)調和同步機制,多個應用程序可以共享一個隊列,并按照先進先出的順序處理隊列中的消息,需要的朋友可以參考下
    2023-08-08

最新評論