在你的应用中添加关闭钩子

概述

启动一个服务通常很容易. 但是, 有时我们需要在停止应用之前执行一个钩子函数,以便优雅的关闭我们的应用. 本文中我们将研究JVM应用程序终止的不同方式. 然后, 我们将使用 Java APIs 管理 JVM shutdown hooks. 在本文中,我们将研究JVM应用程序可以终止的不同方式。 然后,我们将使用Java API管理JVM关闭挂钩。

JVM Shutdown

可以通过两种不同的方式关闭JVM:

1、正常退出

2、异常退出

在以下任一情况下,程序将正常退出JVM:

  • 最后一个非守护程序线程终止. 例如, 当main 线程退出时, the JVM 开始其关闭过程.

  • 从操作系统发送中断信号。 例如,通过按 Ctrl + C 或注销操作系统

  • Java代码调用System.exit()

尽管我们努力实现正常关闭,但有时JVM可能会以突然而意外的方式关闭. 在以下情况下, JVM突然意外关闭:

  • 从操作系统发送终止信号. 例如, 通过执行 kill -9 <jvm_pid>

  • Java代码调用Runtime.getRuntime().halt()

  • 主机OS意外宕机, 例如, 电源故障或其他系统紧急故障

Shutdown Hooks

JVM 允许在关闭之前执行其注册的功能. 这些功能通常用来释放资源或其他类似的任务. 在JVM 术语中, 这些功能被称之为 hutdown hooks . Shutdown hooks 基本上是已初始化但未启动的线程. 当开始其关闭过程时, 它将启动所有已注册的hooks(无顺序性). 运行玩所有钩子程序, JVM然后停止其进程.

添加 Hooks

想要增加一个shutdown hook, 我们可以使用 Runtime.getRuntime().addShutdownHook() 方法:

Thread printingHook = new Thread(() -> System.out.println("In the middle of a shutdown"));
Runtime.getRuntime().addShutdownHook(printingHook);

在这里,我们只是在JVM关闭之前将一些内容打印到标准输出中. 如果我们像下面那样关闭JVM:

> System.exit(129);
In the middle of a shutdown

然后,我们将看到该钩子实际上将消息打印到标准输出。

JVM负责启动hook线程. 因此, 如果给定的hook 已经启动, 则Java将引发异常:

Thread longRunningHook = new Thread(() -> {
    try {
        Thread.sleep(300);
    } catch (InterruptedException ignored) {}
});
longRunningHook.start();
 
assertThatThrownBy(() -> Runtime.getRuntime().addShutdownHook(longRunningHook))
  .isInstanceOf(IllegalArgumentException.class)
  .hasMessage("Hook already running");

显然,我们也不能多次注册同一个钩子:

Thread unfortunateHook = new Thread(() -> {});
Runtime.getRuntime().addShutdownHook(unfortunateHook);
 
assertThatThrownBy(() -> Runtime.getRuntime().addShutdownHook(unfortunateHook))
  .isInstanceOf(IllegalArgumentException.class)
  .hasMessage("Hook previously registered");

删除Hooks

Java 提供了一种孪生的remove 方法,用于在注册后删除特定的 shutdown hook :

Thread willNotRun = new Thread(() -> System.out.println("Won't run!"));
Runtime.getRuntime().addShutdownHook(willNotRun);
 
assertThat(Runtime.getRuntime().removeShutdownHook(willNotRun)).isTrue();

removeShutdownHook() 方法在成功删除shutdown hook 后返回true .

注意事项

JVM 仅在正常终止的情况下运行shutdown hooks . 因此, 当外部突然强制杀死 JVM 进程时, JVM 没有机会执行shutdown hooks. 另外, 使用Java代码关停JVM也将具有相同的效果(不会执行shutdown hooks):

Thread haltedHook = new Thread(() -> System.out.println("Halted abruptly"));
Runtime.getRuntime().addShutdownHook(haltedHook);
         
Runtime.getRuntime().halt(129);

halt 方法强制终止当前正在运行的 JVM. 因此, 已注册的shutdown hooks 将无法被执行.

结语

在本文中, 我们研究了JVM应用程序可能被终止的不同方式. 然后, 我们使用了常用运行时 APIs 来注册和注销 shutdown hooks. 和以前一样, 代码已经上传至 GitHub.

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352