日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

Java ShutdownHook原理詳解

瀏覽:233日期:2022-08-13 15:14:32
ShutdownHook介紹

在java程序中,很容易在進程結(jié)束時添加一個鉤子,即ShutdownHook。通常在程序啟動時加入以下代碼即可

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

有了ShutdownHook我們可以

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

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

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

ShutdownHook原理ShutdownHook的數(shù)據(jù)結(jié)構(gòu)與執(zhí)行順序 當我們添加一個ShutdownHook時,會調(diào)用ApplicationShutdownHooks.add(hook),往ApplicationShutdownHooks類下的靜態(tài)變量private static IdentityHashMap<Thread, Thread> hooks添加一個hook,hook本身是一個thread對象 ApplicationShutdownHooks類初始化時會把hooks添加到Shutdown的hooks中去,而Shutdown的hooks是系統(tǒng)級的ShutdownHook,并且系統(tǒng)級的ShutdownHook由一個數(shù)組構(gòu)成,只能添加10個 系統(tǒng)級的ShutdownHook調(diào)用了thread類的run方法,所以系統(tǒng)級的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)級的ShutdownHook的add方法是包可見,即我們不能直接調(diào)用它 ApplicationShutdownHooks位于下標1處,且應(yīng)用級的hooks,執(zhí)行時調(diào)用的是thread類的start方法,所以應(yīng)用級的ShutdownHook是異步執(zhí)行的,但會等所有hook執(zhí)行完畢才會退出。

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é)如下:

Java ShutdownHook原理詳解

ShutdownHook觸發(fā)點

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

Shutdown.exit

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

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

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

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

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);// 該信號不能捕獲Signal.handle(new Signal('ABRT'), sh);//Signal.handle(new Signal('KILL'), sh);// 該信號不能捕獲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); }}

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

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

Shutdown.shutdown

這個方法可以看注釋

/* 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. */

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

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

舉個例子,當java進程正常退出時,沒有在代碼中主動結(jié)束進程,也沒有kill,就像這樣

public static void main(String[] args) { Runtime.getRuntime().addShutdownHook(new Thread() {@Overridepublic void run() { super.run(); System.out.println('I’m shutdown hook ');} });}

當main線程運行完了后,也能打印出I’m shutdown hook,反觀golang就做不到這一點(如果可以做到,可以私信告訴我,我是個golang新手)

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

Java ShutdownHook原理詳解

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

總結(jié)

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

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

以上就是Java ShutdownHook原理詳解的詳細內(nèi)容,更多關(guān)于Java ShutdownHook原理的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標簽: Java
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
精品中国亚洲| 国产精品一区二区美女视频免费看| 男女激情视频一区| 久久午夜精品| 亚洲精品麻豆| 国产乱码精品一区二区亚洲| 国产精品白浆| 国产不卡人人| 亚洲精品国产偷自在线观看| 亚洲综合精品四区| 亚洲tv在线| 国产精品免费精品自在线观看| 精品国产一区二区三区av片| 香蕉视频亚洲一级| 亚洲电影在线| 亚洲精品福利| 精品五月天堂| 免费污视频在线一区| 国产农村妇女精品一二区| 日韩高清中文字幕一区| 国产精品国产三级国产在线观看| 四虎影视精品| 视频一区中文字幕精品| 精品中文字幕一区二区三区| 欧美va天堂在线| 日韩二区三区四区| 欧美精品高清| 日韩av午夜在线观看| 正在播放日韩精品| 免费精品视频最新在线| 国产高清视频一区二区| 国产一区清纯| 日本国产一区| 日韩久久精品网| 只有精品亚洲| 亚洲日韩中文字幕一区| 亚洲精品乱码日韩| 国产91在线精品| 日韩中文字幕亚洲一区二区va在线| 国产精品一国产精品| 色婷婷狠狠五月综合天色拍| 亚洲免费观看高清完整版在线观| 国产成人a视频高清在线观看| 美女尤物久久精品| 国产美女高潮在线观看| 日韩中文字幕一区二区高清99| 成人小电影网站| 亚洲九九精品| 99成人在线视频| 国产精品久久久久久模特 | 激情视频网站在线播放色| 在线精品国产亚洲| 天堂中文av在线资源库| 欧美日韩18| 在线 亚洲欧美在线综合一区| 久久精品国产久精国产| 免费观看久久久4p| 在线日韩电影| 国内一区二区三区| 日韩一区二区三区精品 | 麻豆91小视频| 日本不卡一二三区黄网| 99视频精品免费观看| 国产在线观看www| 国产免费av一区二区三区| 亚洲午夜久久| 羞羞答答国产精品www一本| 成人久久一区| 国产精品yjizz视频网| 国产伦理久久久久久妇女| 亚洲女同一区| 亚洲www啪成人一区二区| 久久伊人国产| 欧美国产另类| 国产精品探花在线观看| 日韩黄色av| 一本综合精品| 中文字幕中文字幕精品| 亚洲一区区二区| 欧美福利在线| 亚洲女同一区| 亚洲成人三区| 一区二区小说| 亚洲黑丝一区二区| 群体交乱之放荡娇妻一区二区| 免费一级欧美片在线观看网站| 日韩手机在线| 日韩精品久久久久久| 日韩中文av| 亚洲欧美视频一区二区三区| 伊人成人网在线看| 国产亚洲毛片| 久久都是精品| 国产模特精品视频久久久久| 日韩午夜一区| 视频在线在亚洲| 最近国产精品视频| 日韩区一区二| 久久黄色影视| 国产精品成人一区二区网站软件| 国产精品亚洲片在线播放| 亚州国产精品| 日韩av在线免费观看不卡| 欧美日韩一区二区三区四区在线观看 | 狠狠干成人综合网| 亚洲激情不卡| 羞羞答答国产精品www一本| 六月婷婷一区| 亚洲精品动态| 国产麻豆精品久久| 97精品中文字幕| 999国产精品视频| 91久久黄色| 日韩国产欧美在线播放| 国产精品高清一区二区| av免费不卡国产观看| 午夜国产一区二区| 亚洲欧美网站在线观看| 国产精品羞羞答答在线观看| 成人污污视频| 欧美日韩水蜜桃| 免费成人av在线播放| 国产免费播放一区二区| 国产精品成人**免费视频| 精品久久久网| 亚洲激情中文| 久久国产视频网| 欧美日韩视频免费观看| 香蕉久久国产| 国产精品一站二站| 桃色一区二区| 免费的成人av| 精品中文在线| 尹人成人综合网| 欧美日韩一区二区三区四区在线观看 | 日韩精品电影| 石原莉奈一区二区三区在线观看| 欧美日韩中文| 欧美/亚洲一区| 国产欧美日韩综合一区在线播放| 黄在线观看免费网站ktv| 国产视频亚洲| 免费亚洲婷婷| 夜久久久久久| 精品入口麻豆88视频| 99国产精品99久久久久久粉嫩| 国产精品中文字幕亚洲欧美| 亚洲欧洲高清| 久久国产人妖系列| 一区在线免费| 精品72久久久久中文字幕| 日韩精品一级二级| 日韩成人a**站| 欧美永久精品| 亚洲综合丁香| 在线天堂中文资源最新版| 日韩三区四区| 亚洲成人精选| 国产高清不卡| 国产亚洲一区二区三区啪| 夜夜精品视频| 日韩免费久久| 青青伊人久久| 亚洲视频www| 国产aa精品| 国产精品色在线网站| 视频精品一区二区| 久久久久.com| 国产精品福利在线观看播放| 日本少妇精品亚洲第一区| 波多野结衣一区| 91综合网人人| 国产精品欧美在线观看| 丝瓜av网站精品一区二区| 中文字幕在线视频久| 久久av影院| 日韩av资源网| 久久电影一区| 欧美99久久| se01亚洲视频| 国内精品美女在线观看| 国产日韩免费| 亚洲精品第一| 视频在线在亚洲| 日韩av网站在线免费观看| 国产视频欧美| 国产综合精品| 五月激情久久| 精品美女视频 | 日韩有吗在线观看| 亚洲性色视频| 999精品色在线播放| 高清精品久久| 麻豆视频一区二区| 国产精品男女| 国产日本久久| 国产日韩一区二区三免费高清| 日韩中文av| 日韩av中文字幕一区| 日韩视频一二区|