聊聊如何优雅的关闭服务?

开发 前端
JVM 只在正常终止的情况下运行关闭钩子。因此,当外部力量突然杀死JVM进程时,JVM将没有机会执行关闭钩子。此外,从Java代码中停止JVM也会产生同样的效果。

大家好,我是指北君。

通常,启动一个服务是很容易的。然而,有时我们需要有一个计划来优雅地关闭一个服务。

在本教程中,我们将看一下 JVM 应用程序终止的不同方式。然后,我们将使用 Java APIs 来管理 JVM 关闭钩子。

关闭 JVM

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

  • 一种受控的方式
  • 一种非受控的方式

一个受控的进程在以下两种情况下关闭 JVM。

  • 最后一个非 daemon 线程终止。例如,当主线程退出时,JVM 开始其关闭进程
  • 从操作系统发送一个中断信号。例如,通过按 Ctrl + C 或注销操作系统
  • 从 Java 代码中调用 System.exit()

虽然我们都在努力争取优雅的关闭,但有时 JVM 可能会以突然和意外的方式关闭。JVM 会在以下情况下突然关闭。

  • 从操作系统发送一个 kill 信号。例如,通过发出 kill -9 的信号
  • 从 Java 代码中调用 Runtime.getRuntime().halt() 。
  • 主机操作系统意外关闭,例如,在电源故障或操作系统崩溃的情况下

shutdown hook

JVM 允许在完成关机之前运行注册函数。这些函数通常是释放资源或其他类似的内部管理任务的好地方。在 JVM 的术语中,这些函数被称为关闭钩子。

关闭钩子基本上是初始化但未启动的线程。当JVM开始其关闭过程时,它将以一个未指定的顺序启动所有注册的钩子。在运行完所有钩子后,JVM 将停止运行。

添加钩子

为了添加一个关闭钩子,我们可以使用 Runtime.getRuntime().addShutdownHook() 方法。

Thread printingHook = new Thread(() -> System.out.println("我要关闭了"));
Runtime.getRuntime().addShutdownHook(printingHook);

在这里,我们只是在JVM自行关闭之前向标准输出端打印一些东西。如果我们像下面这样关闭JVM。

System.exit(123);

我要关闭了

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

JVM负责启动钩子线程。因此,如果给定的钩子已经被启动了,Java将抛出一个异常。

Thread longRunningHook = new Thread(() -> {
try {
Thread.sleep(300);
} catch (InterruptedException ignored) {}
});
longRunningHook.start();

assertThatThrownBy(() -> Runtime.getRuntime().addShutdownHook(longRunningHook))
.isInstanceOf(IllegalArgumentException.class)
.hasMessage("钩子正在运行");

很明显,我们也不能多次注册一个钩子。

Thread unfortunateHook = new Thread(() -> {});
Runtime.getRuntime().addShutdownHook(unfortunateHook);

assertThatThrownBy(() -> Runtime.getRuntime().addShutdownHook(unfortunateHook))
.isInstanceOf(IllegalArgumentException.class)
.hasMessage("钩子已经注册");

删除钩子

Java 提供了一个孪生的移除方法,以便在注册一个特定的关闭钩子后将其移除。

Thread willNotRun = new Thread(() -> System.out.println("钩子不会运行的"));
Runtime.getRuntime().addShutdownHook(willNotRun);

assertThat(Runtime.getRuntime().removeShutdownHook(willNotRun)).isTrue();

当关闭钩子被成功删除时,removeShutdownHook() 方法返回true。

注意事项

JVM 只在正常终止的情况下运行关闭钩子。因此,当外部力量突然杀死JVM进程时,JVM将没有机会执行关闭钩子。此外,从Java代码中停止JVM也会产生同样的效果。

Thread haltedHook = new Thread(() -> System.out.println("强行终止"));
Runtime.getRuntime().addShutdownHook(haltedHook);

Runtime.getRuntime().halt(123);

halt 方法强行终止了当前运行的JVM。因此,注册的关闭钩子不会有机会执行。

总结

在本教程中,我们研究了 JVM 应用程序可能终止的不同方式。然后,我们使用一些运行时API来注册和取消注册关闭钩子。

责任编辑:武晓燕 来源: Java技术指北
相关推荐

2021-01-19 10:35:49

JVM场景函数

2021-03-28 09:17:18

JVM场景钩子函数

2017-12-19 10:03:44

JavaLinux代码

2021-12-06 09:57:25

容器Linux信号

2020-10-16 11:48:06

服务器系统运维

2021-04-20 08:00:31

Redisson关闭订单支付系统

2022-09-08 07:32:56

JDK7程序管理

2019-11-18 15:50:11

AjaxJavascript前端

2021-07-14 06:45:49

Windows.NetTopshelf

2023-11-30 07:40:05

URLCMS

2015-11-26 10:53:45

LinuxWindowsMac OS

2017-07-26 11:32:50

NETRabbitMQ系统集成

2018-01-12 15:59:36

2023-12-20 10:04:45

线程池Java

2020-11-06 08:13:03

服务器Nodejs客户端

2022-11-08 08:35:53

架构微服务移动

2023-01-29 09:06:24

微服务划分关联

2020-08-26 07:17:19

通信

2023-10-10 13:23:18

空指针异常Java

2023-10-19 19:42:25

IstioPodkubernetes
点赞
收藏

51CTO技术栈公众号