亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java ShutdownHook原理詳解

 更新時(shí)間:2021年04月26日 09:59:58   作者:捉蟲大師  
這篇文章主要介紹了Java ShutdownHook原理的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)使用Java,感興趣的朋友可以了解下

ShutdownHook介紹

在java程序中,很容易在進(jìn)程結(jié)束時(shí)添加一個(gè)鉤子,即ShutdownHook。通常在程序啟動(dòng)時(shí)加入以下代碼即可

Runtime.getRuntime().addShutdownHook(new Thread(){
    @Override
    public void run() {
        System.out.println("I'm shutdown hook...");
    }
});

有了ShutdownHook我們可以

  • 在進(jìn)程結(jié)束時(shí)做一些善后工作,例如釋放占用的資源,保存程序狀態(tài)等
  • 為優(yōu)雅(平滑)發(fā)布提供手段,在程序關(guān)閉前摘除流量

不少java中間件或框架都使用了ShutdownHook的能力,如dubbo、spring等。

spring中在application context被load時(shí)會(huì)注冊(cè)一個(gè)ShutdownHook。 這個(gè)ShutdownHook會(huì)在進(jìn)程退出前執(zhí)行銷毀bean,發(fā)出ContextClosedEvent等動(dòng)作。 而dubbo在spring框架下正是監(jiān)聽了ContextClosedEvent,調(diào)用dubboBootstrap.stop()來實(shí)現(xiàn)清理現(xiàn)場和dubbo的優(yōu)雅發(fā)布,spring的事件機(jī)制默認(rèn)是同步的,所以能在publish事件時(shí)等待所有監(jiān)聽者執(zhí)行完畢。

ShutdownHook原理

ShutdownHook的數(shù)據(jù)結(jié)構(gòu)與執(zhí)行順序

  • 當(dāng)我們添加一個(gè)ShutdownHook時(shí),會(huì)調(diào)用ApplicationShutdownHooks.add(hook),往ApplicationShutdownHooks類下的靜態(tài)變量private static IdentityHashMap<Thread, Thread> hooks添加一個(gè)hook,hook本身是一個(gè)thread對(duì)象
  • ApplicationShutdownHooks類初始化時(shí)會(huì)把hooks添加到Shutdown的hooks中去,而Shutdown的hooks是系統(tǒng)級(jí)的ShutdownHook,并且系統(tǒng)級(jí)的ShutdownHook由一個(gè)數(shù)組構(gòu)成,只能添加10個(gè)
  • 系統(tǒng)級(jí)的ShutdownHook調(diào)用了thread類的run方法,所以系統(tǒng)級(jí)的ShutdownHook是同步有序執(zhí)行的
private static void runHooks() {
    for (int i=0; i < MAX_SYSTEM_HOOKS; i++) {
        try {
            Runnable hook;
            synchronized (lock) {
                // acquire the lock to make sure the hook registered during
                // shutdown is visible here.
                currentRunningHook = i;
                hook = hooks[i];
            }
            if (hook != null) hook.run();
        } catch(Throwable t) {
            if (t instanceof ThreadDeath) {
                ThreadDeath td = (ThreadDeath)t;
                throw td;
            }
        }
    }
}
  • 系統(tǒng)級(jí)的ShutdownHook的add方法是包可見,即我們不能直接調(diào)用它
  • ApplicationShutdownHooks位于下標(biāo)1處,且應(yīng)用級(jí)的hooks,執(zhí)行時(shí)調(diào)用的是thread類的start方法,所以應(yīng)用級(jí)的ShutdownHook是異步執(zhí)行的,但會(huì)等所有hook執(zhí)行完畢才會(huì)退出。
static void runHooks() {
    Collection<Thread> threads;
    synchronized(ApplicationShutdownHooks.class) {
        threads = hooks.keySet();
        hooks = null;
    }

    for (Thread hook : threads) {
        hook.start();
    }
    for (Thread hook : threads) {
        while (true) {
            try {
                hook.join();
                break;
            } catch (InterruptedException ignored) {
            }
        }
    }
}

用一副圖總結(jié)如下:

ShutdownHook觸發(fā)點(diǎn)

從Shutdown的runHooks順藤摸瓜,我們得出以下這個(gè)調(diào)用路徑

Shutdown.exit

跟進(jìn)Shutdown.exit的調(diào)用方,發(fā)現(xiàn)有 Runtime.exit 和 Terminator.setup

  • Runtime.exit 是代碼中主動(dòng)結(jié)束進(jìn)程的接口
  • Terminator.setup 被 initializeSystemClass 調(diào)用,當(dāng)?shù)谝粋€(gè)線程被初始化的時(shí)候被觸發(fā),觸發(fā)后注冊(cè)了一個(gè)信號(hào)監(jiān)控函數(shù),捕獲kill發(fā)出的信號(hào),調(diào)用Shutdown.exit結(jié)束進(jìn)程

這樣覆蓋了代碼中主動(dòng)結(jié)束進(jìn)程和被kill殺死進(jìn)程的場景。

主動(dòng)結(jié)束進(jìn)程不必介紹,這里說一下信號(hào)捕獲。在java中我們可以寫出如下代碼來捕獲kill信號(hào),只需要實(shí)現(xiàn)SignalHandler接口以及handle方法,程序入口處注冊(cè)要監(jiān)聽的相應(yīng)信號(hào)即可,當(dāng)然不是每個(gè)信號(hào)都能捕獲處理。

public class SignalHandlerTest implements SignalHandler {

    public static void main(String[] args) {

        Runtime.getRuntime().addShutdownHook(new Thread() {
            @Override
            public void run() {
                System.out.println("I'm shutdown hook ");
            }
        });

        SignalHandler sh = new SignalHandlerTest();
        Signal.handle(new Signal("HUP"), sh);
        Signal.handle(new Signal("INT"), sh);
        //Signal.handle(new Signal("QUIT"), sh);// 該信號(hào)不能捕獲
        Signal.handle(new Signal("ABRT"), sh);
        //Signal.handle(new Signal("KILL"), sh);// 該信號(hào)不能捕獲
        Signal.handle(new Signal("ALRM"), sh);
        Signal.handle(new Signal("TERM"), sh);

        while (true) {
            System.out.println("main running");
            try {
                Thread.sleep(2000L);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    @Override
    public void handle(Signal signal) {
        System.out.println("receive signal " + signal.getName() + "-" + signal.getNumber());
        System.exit(0);
    }
}

要注意的是通常來說,我們捕獲信號(hào),做了一些個(gè)性化的處理后需要主動(dòng)調(diào)用System.exit,否則進(jìn)程就不會(huì)退出了,這時(shí)只能使用kill -9來強(qiáng)制殺死進(jìn)程了。

而且每次信號(hào)的捕獲是在不同的線程中,所以他們之間的執(zhí)行是異步的。

Shutdown.shutdown

這個(gè)方法可以看注釋

/* Invoked by the JNI DestroyJavaVM procedure when the last non-daemon
  * thread has finished.  Unlike the exit method, this method does not
  * actually halt the VM.
  */

翻譯一下就是該方法會(huì)在最后一個(gè)非daemon線程(非守護(hù)線程)結(jié)束時(shí)被JNI的DestroyJavaVM方法調(diào)用。

java中有兩類線程,用戶線程和守護(hù)線程,守護(hù)線程是服務(wù)于用戶線程,如GC線程,JVM判斷是否結(jié)束的標(biāo)志就是是否還有用戶線程在工作。 當(dāng)最后一個(gè)用戶線程結(jié)束時(shí),就會(huì)調(diào)用 Shutdown.shutdown。這是JVM這類虛擬機(jī)語言特有的"權(quán)利",倘若是golang這類編譯成可執(zhí)行的二進(jìn)制文件時(shí),當(dāng)全部用戶線程結(jié)束時(shí)是不會(huì)執(zhí)行ShutdownHook的。

舉個(gè)例子,當(dāng)java進(jìn)程正常退出時(shí),沒有在代碼中主動(dòng)結(jié)束進(jìn)程,也沒有kill,就像這樣

public static void main(String[] args) {

    Runtime.getRuntime().addShutdownHook(new Thread() {
        @Override
        public void run() {
            super.run();
            System.out.println("I'm shutdown hook ");
        }
    });
}

當(dāng)main線程運(yùn)行完了后,也能打印出I'm shutdown hook,反觀golang就做不到這一點(diǎn)(如果可以做到,可以私信告訴我,我是個(gè)golang新手)

通過如上兩個(gè)調(diào)用的分析,我們概括出如下結(jié)論:

我們能看出java的ShutdownHook其實(shí)覆蓋的非常全面了,只有一處無法覆蓋,即當(dāng)我們殺死進(jìn)程時(shí)使用了kill -9時(shí),由于程序無法捕獲處理,進(jìn)程被直接殺死,所以無法執(zhí)行ShutdownHook。

總結(jié)

綜上,我們得出一些結(jié)論

  • 重寫捕獲信號(hào)需要注意主動(dòng)退出進(jìn)程,否則進(jìn)程可能永遠(yuǎn)不會(huì)退出,捕獲信號(hào)的執(zhí)行是異步的
  • 用戶級(jí)的ShutdownHook是綁定在系統(tǒng)級(jí)的ShutdownHook之上,且用戶級(jí)是異步執(zhí)行,系統(tǒng)級(jí)是同步順序執(zhí)行,用戶級(jí)處于系統(tǒng)級(jí)執(zhí)行順序的第二位
  • ShutdownHook 覆蓋的面比較廣,不論是手動(dòng)調(diào)用接口退出進(jìn)程,還是捕獲信號(hào)退出進(jìn)程,抑或是用戶線程執(zhí)行完畢退出,都會(huì)執(zhí)行ShutdownHook,唯一不會(huì)執(zhí)行的就是kill -9

以上就是Java ShutdownHook原理詳解的詳細(xì)內(nèi)容,更多關(guān)于Java ShutdownHook原理的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Spring Cloud Alibaba使用Sentinel實(shí)現(xiàn)接口限流

    Spring Cloud Alibaba使用Sentinel實(shí)現(xiàn)接口限流

    這篇文章主要介紹了Spring Cloud Alibaba使用Sentinel實(shí)現(xiàn)接口限流,本文詳細(xì)的介紹了Sentinel組件的用法以及接口限流,感興趣的可以了解一下
    2019-04-04
  • 深入分析RabbitMQ中死信隊(duì)列與死信交換機(jī)

    深入分析RabbitMQ中死信隊(duì)列與死信交換機(jī)

    這篇文章主要介紹了RabbitMQ中死信隊(duì)列與死信交換機(jī),死信隊(duì)列就是一個(gè)普通的交換機(jī),有些隊(duì)列的消息成為死信后,一般情況下會(huì)被RabbitMQ清理,感興趣想要詳細(xì)了解可以參考下文
    2023-05-05
  • 5種Java經(jīng)典創(chuàng)建型模式詳解

    5種Java經(jīng)典創(chuàng)建型模式詳解

    這篇文章主要為大家詳細(xì)介紹了5種Java經(jīng)典創(chuàng)建型模式,感興趣的小伙伴們可以參考一下
    2016-03-03
  • java基礎(chǔ)-給出一個(gè)隨機(jī)字符串,判斷有多少字母?多少數(shù)字?

    java基礎(chǔ)-給出一個(gè)隨機(jī)字符串,判斷有多少字母?多少數(shù)字?

    這篇文章主要介紹了java基礎(chǔ)-給出一個(gè)隨機(jī)字符串,判斷有多少字母?多少數(shù)字?文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • 詳解maven的setting配置文件中mirror和repository的區(qū)別

    詳解maven的setting配置文件中mirror和repository的區(qū)別

    這篇文章主要介紹了詳解maven的setting配置文件中mirror和repository的區(qū)別,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-12-12
  • java中transient關(guān)鍵字的作用解析

    java中transient關(guān)鍵字的作用解析

    這篇文章主要介紹了java中transient關(guān)鍵字的作用解析,日常業(yè)務(wù)中,為了安全起見,有些敏感信息我們不希望在網(wǎng)絡(luò)間被傳輸可以使用transient對(duì)字段進(jìn)行修飾,不進(jìn)行序列化,則返回獲取到的字段為null,需要的朋友可以參考下
    2023-11-11
  • 基于Java設(shè)計(jì)一個(gè)高并發(fā)的秒殺系統(tǒng)

    基于Java設(shè)計(jì)一個(gè)高并發(fā)的秒殺系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了如何基于Java設(shè)計(jì)一個(gè)高并發(fā)的秒殺系統(tǒng),文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,有需要的小伙伴可以參考下
    2023-10-10
  • Java向Runnable線程傳遞參數(shù)方法實(shí)例解析

    Java向Runnable線程傳遞參數(shù)方法實(shí)例解析

    這篇文章主要介紹了Java向Runnable線程傳遞參數(shù)方法實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-06-06
  • spring初始化源碼代碼淺析

    spring初始化源碼代碼淺析

    Spring框架被廣泛應(yīng)用于我們的日常工作中,但是很長時(shí)間以來我們都是只會(huì)使用,不懂它的作用原理,下面這篇文章主要給大家介紹了關(guān)于spring初始化源碼的相關(guān)資料,需要的朋友可以參考下
    2023-04-04
  • Java重寫(Override)與重載(Overload)區(qū)別原理解析

    Java重寫(Override)與重載(Overload)區(qū)別原理解析

    這篇文章主要介紹了Java重寫(Override)與重載(Overload)區(qū)別原理解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-02-02

最新評(píng)論