电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

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

Java ShutdownHook原理詳解

瀏覽:116日期: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í)行順序 當(dāng)我們添加一個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位于下標(biāo)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)用,當(dāng)?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)信號即可,當(dā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é)束的標(biāo)志就是是否還有用戶線程在工作。 當(dāng)最后一個用戶線程結(jié)束時,就會調(diào)用 Shutdown.shutdown。這是JVM這類虛擬機語言特有的'權(quán)利',倘若是golang這類編譯成可執(zhí)行的二進制文件時,當(dāng)全部用戶線程結(jié)束時是不會執(zhí)行ShutdownHook的。

舉個例子,當(dāng)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 ');} });}

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

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

Java ShutdownHook原理詳解

我們能看出java的ShutdownHook其實覆蓋的非常全面了,只有一處無法覆蓋,即當(dāng)我們殺死進程時使用了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)文章!

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 东亚液氮罐-液氮生物容器-乐山市东亚机电工贸有限公司 | 客服外包专业服务商_客服外包中心_网萌科技 | 中直网_行业门户-行业人专业的交流平台!| 达利园物流科技集团- | 滑石粉,滑石粉厂家,超细滑石粉-莱州圣凯滑石有限公司 | 活性氧化铝球|氧化铝干燥剂|分子筛干燥剂|氢氧化铝粉-淄博同心材料有限公司 | 上海办公室装修,办公楼装修设计,办公空间设计,企业展厅设计_写艺装饰公司 | LED显示屏_LED屏方案设计精准报价专业安装丨四川诺显科技 | 东莞螺杆空压机_永磁变频空压机_节能空压机_空压机工厂批发_深圳螺杆空压机_广州螺杆空压机_东莞空压机_空压机批发_东莞空压机工厂批发_东莞市文颖设备科技有限公司 | 卫生纸复卷机|抽纸机|卫生纸加工设备|做卫生纸机器|小型卫生纸加工需要什么设备|卫生纸机器设备多少钱一台|许昌恒源纸品机械有限公司 | 铝合金线槽_铝型材加工_空调挡水板厂家-江阴炜福金属制品有限公司 | 天津蒸汽/热水锅炉-电锅炉安装维修直销厂家-天津鑫淼暖通设备有限公司 | 苏州同创电子有限公司 - 四探针测试仪源头厂家 | 防爆电机_防爆电机型号_河南省南洋防爆电机有限公司 | 汽液过滤网厂家_安平县银锐丝网有限公司 | 不锈钢复合板厂家_钛钢复合板批发_铜铝复合板供应-威海泓方金属复合材料股份有限公司 | 翰香原枣子坊加盟费多少钱-正宗枣核糕配方培训利润高飘香 | 慈溪麦田广告公司,提供慈溪广告设计。| 耐磨焊丝,堆焊焊丝,耐磨药芯焊丝,碳化钨焊丝-北京耐默公司 | 双工位钻铣攻牙机-转换工作台钻攻中心-钻铣攻牙机一体机-浙江利硕自动化设备有限公司 | 滚塑PE壳体-PE塑料浮球-警示PE浮筒-宁波君益塑业有限公司 | 上海办公室装修,办公楼装修设计,办公空间设计,企业展厅设计_写艺装饰公司 | 热回收盐水机组-反应釜冷水机组-高低温冷水机组-北京蓝海神骏科技有限公司 | 拼装地板,悬浮地板厂家,悬浮式拼装运动地板-石家庄博超地板科技有限公司 | 臻知网大型互动问答社区-你的问题将在这里得到解答!-无锡据风网络科技有限公司 | 环讯传媒,永康网络公司,永康网站建设,永康小程序开发制作,永康网站制作,武义网页设计,金华地区网站SEO优化推广 - 永康市环讯电子商务有限公司 | 专业深孔加工_东莞深孔钻加工_东莞深孔钻_东莞深孔加工_模具深孔钻加工厂-东莞市超耀实业有限公司 | 微学堂-电动能源汽车评测_电动车性能分享网| 石油/泥浆/不锈钢防腐/砂泵/抽砂泵/砂砾泵/吸砂泵/压滤机泵 - 专业石油环保专用泵厂家 | 利浦顿蒸汽发生器厂家-电蒸汽发生器/燃气蒸汽发生器_湖北利浦顿热能科技有限公司官网 | 托利多电子平台秤-高精度接线盒-托利多高精度电子秤|百科 | 创富网-B2B网站|供求信息网|b2b平台|专业电子商务网站 | 广州工业氧气-工业氩气-工业氮气-二氧化碳-广州市番禺区得力气体经营部 | 天津拓展_天津团建_天津趣味运动会_天津活动策划公司-天津华天拓展培训中心 | 环氧铁红防锈漆_环氧漆_无溶剂环氧涂料_环氧防腐漆-华川涂料 | 网优资讯-为循环资源、大宗商品、工业服务提供资讯与行情分析的数据服务平台 | 液压油缸-液压缸厂家价格,液压站系统-山东国立液压制造有限公司 液压油缸生产厂家-山东液压站-济南捷兴液压机电设备有限公司 | 科普仪器菏泽市教育教学仪器总厂 | 流水线电子称-钰恒-上下限报警电子秤-上海宿衡实业有限公司 | 分光色差仪,测色仪,反透射灯箱,爱色丽分光光度仪,美能达色差仪维修_苏州欣美和仪器有限公司 | ERP企业管理系统永久免费版_在线ERP系统_OA办公_云版软件官网 |