常见Java应用如何优雅关闭

发布于 2017-12-20 | 作者: 51Testing软件测试网 | 来源: 网易号 | 转载于: 网易号

一、前言

在我们进行系统升级的时候,往往需要关闭我们的应用,然后重启。在关闭应用前,我们希望做一些前置操作,比如关闭数据库、redis连接,清理zookeeper的临时节点,释放分布式锁,持久化缓存数据等等。

二、Linux的信号机制

在linux上,我们关闭进程主要是使用kill <pid>的方式。

当执行该命令以后,linux会向进程发送一个信号,进程收到以后之后,可以做一些清理工作。

kill命令默认的信号值为15,即SIGTERM信号。

通过kill -l查看linux支持哪些信号:

linux提供了signal()api,可以将信号处理函数注册上去:

三、Java提供的Shutdown Hook

Java并不支持类似于linux的信号机制,但是提供了Runtime.addShutdownHook(Thread hook)的api。

在JVM关闭前,会并发执行各个Hook线程。

四、Spring Boot提供的优雅关闭功能

我们一般采用如下的方式,启动一个Spring boot应用:

SpringApplication.run()代码如下,会调用到refreshContext(context)方法:

refreshContext()方法比较简单:

很明显,Spring boot通过在启动时,向JVM注册一个ShutdownHook,从而实现JVM关闭前,正常关闭Spring容器。而Spring在销毁时,会依次调用bean的destroy动作来销毁。

五、Dubbo的优雅关闭策略

Dubbo同样是基于ShutdownHook实现的。

AbstractConfig的static代码:

六、总结

只要我们的应用运行在linux平台上,所有的优雅关闭方案都是基于linux提供的信号机制提供的,JVM也是如此。

Java并没有为我们提供与之一一对应的api,而是给出了个ShutdownHook机制,也能达到类似的效果,缺点是我们无法得知JVM关闭的原因。

像dubbo、spring boot等成熟的开源框架,都实现了自动注册ShutdownHook的功能,从而避免使用者忘记调用优雅关闭api引发问题,降低框架的使用难度。

更多JAVA相关文章,请前往51Ttesting软件测试网链接