Spring Boot 優雅停止服務的幾種方法
作者 | 黃青石
來源 | https://www.cnblogs.com/huangqingshi/p/11370291.html
最近突然想到了優雅停止 SpringBoot 服務問題,在使用 SpringBoot 的時候,都要涉及到服務的停止和啓動,當我們停止服務的時候,很多時候大家都是 kill -9 直接把程序進程殺掉,這樣程序不會執行優雅的關閉。而且一些沒有執行完的程序就會直接退出。
我們很多時候都需要安全的將服務停止,也就是把沒有處理完的工作繼續處理完成。比如停止一些依賴的服務,輸出一些日誌,發一些信號給其他的應用系統,這個在保證系統的高可用是非常有必要的。那麼咱麼就來看一下幾種停止 SpringBoot 的方法。
第一種
第一種就是 Springboot 提供的 actuator 的功能,它可以執行 shutdown, health, info 等,默認情況下,actuator 的 shutdown 是 disable 的,我們需要打開它。首先引入 acturator 的 maven 依賴。
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
然後將 shutdown 節點打開,也將 / actuator/shutdown 暴露 web 訪問也設置上,除了 shutdown 之外還有 health, info 的 web 訪問都打開的話將 management.endpoints.web.exposure.include=* 就可以。將如下配置設置到 application.properties 裏邊。設置一下服務的端口號爲 3333。
server.port=3333
management.endpoint.shutdown.enabled=true
management.endpoints.web.exposure.include=shutdown
接下來,咱們創建一個 springboot 工程,然後設置一個 bean 對象,配置上 PreDestroy 方法。這樣在停止的時候會打印語句。bean 的整個生命週期分爲創建、初始化、銷燬,當最後關閉的時候會執行銷燬操作。在銷燬的方法中執行一條輸出日誌。
package com.hqs.springboot.shutdowndemo.bean;
import javax.annotation.PreDestroy;
/**
* @author huangqingshi
* @Date 2019-08-17
*/
public class TerminateBean {
@PreDestroy
public void preDestroy() {
System.out.println("TerminalBean is destroyed");
}
}
做一個 configuration,然後提供一個獲取 bean 的方法,這樣該 bean 對象會被初始化。
package com.hqs.springboot.shutdowndemo.config;
import com.hqs.springboot.shutdowndemo.bean.TerminateBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
* @author huangqingshi
* @Date 2019-08-17
*/
@Configuration
public class ShutDownConfig {
@Bean
public TerminateBean getTerminateBean() {
return new TerminateBean();
}
}
在啓動類裏邊輸出一個啓動日誌,當工程啓動的時候,會看到啓動的輸出,接下來咱們執行停止命令。
curl -X POST http://localhost:3333/actuator/shutdown
以下日誌可以輸出啓動時的日誌打印和停止時的日誌打印,同時程序已經停止。是不是比較神奇。
4 月程序員工資統計出爐:平均 14596 元,南京程序員收入擠進一線。
第二種
第二種方法也比較簡單,獲取程序啓動時候的 context,然後關閉主程序啓動時的 context。這樣程序在關閉的時候也會調用 PreDestroy 註解。如下方法在程序啓動十秒後進行關閉。
/* method 2: use ctx.close to shutdown all application context */
ConfigurableApplicationContext ctx = SpringApplication.run(ShutdowndemoApplication.class, args);
try {
TimeUnit.SECONDS.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
ctx.close();
第三種
第三種方法,在 springboot 啓動的時候將進程號寫入一個 app.pid 文件,生成的路徑是可以指定的,可以通過命令 cat /Users/huangqingshi/app.id | xargs kill 命令直接停止服務,這個時候 bean 對象的 PreDestroy 方法也會調用的。這種方法大家使用的比較普遍。寫一個 start.sh 用於啓動 springboot 程序,然後寫一個停止程序將服務停止。
/* method 3 : generate a pid in a specified path, while use command to shutdown pid :
'cat /Users/huangqingshi/app.pid | xargs kill' */
SpringApplication application = new SpringApplication(ShutdowndemoApplication.class);
application.addListeners(new ApplicationPidFileWriter("/Users/huangqingshi/app.pid"));
application.run();
第四種
第四種方法,通過調用一個 SpringApplication.exit()方法也可以退出程序,同時將生成一個退出碼,這個退出碼可以傳遞給所有的 context。這個就是一個 JVM 的鉤子,通過調用這個方法的話會把所有 PreDestroy 的方法執行並停止,並且傳遞給具體的退出碼給所有 Context。通過調用 System.exit(exitCode) 可以將這個錯誤碼也傳給 JVM。程序執行完後最後會輸出:Process finished with exit code 0,給 JVM 一個 SIGNAL。
/* method 4: exit this application using static method */
ConfigurableApplicationContext ctx = SpringApplication.run(ShutdowndemoApplication.class, args);
exitApplication(ctx);
public static void exitApplication(ConfigurableApplicationContext context) {
int exitCode = SpringApplication.exit(context, (ExitCodeGenerator) () -> 0);
System.exit(exitCode);
}
第五種
第五種方法,自己寫一個 Controller,然後將自己寫好的 Controller 獲取到程序的 context,然後調用自己配置的 Controller 方法退出程序。通過調用自己寫的 / shutDownContext 方法關閉程序:curl -X POST http://localhost:3333/shutDownContext。
package com.hqs.springboot.shutdowndemo.controller;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
/**
* @author huangqingshi
* @Date 2019-08-17
*/
@RestController
public class ShutDownController implements ApplicationContextAware {
private ApplicationContext context;
@PostMapping("/shutDownContext")
public String shutDownContext() {
ConfigurableApplicationContext ctx = (ConfigurableApplicationContext) context;
ctx.close();
return "context is shutdown";
}
@GetMapping("/")
public String getIndex() {
return "OK";
}
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
context = applicationContext;
}
}
好了,springboot 的優雅關閉方法也都實現好了,也有同學問,如何暴力停止呢,簡單,直接 kill -9 相應的 PID 即可。
總結一下
以上這幾種方法實現的話比較簡單,但是真實工作中還需要考慮的點還很多,比如需要保護暴露的點不被別人利用,一般要加一些防火牆,或者只在內網使用,保證程序安全。
在真實的工作中的時候第三種比較常用,程序中一般使用內存隊列或線程池的時候最好要優雅的關機,將內存隊列沒有處理的保存起來或線程池中沒處理完的程序處理完。但是因爲停機的時候比較快,所以停服務的時候最好不要處理大量的數據操作,這樣會影響程序停止。
好了,大家覺得還沒看全的話,可以訪問我的 GIT 代碼:
https://github.com/stonehqs/shutdowndemo.git 。
本文由 Readfog 進行 AMP 轉碼,版權歸原作者所有。
來源:https://mp.weixin.qq.com/s/0Hc-S44IjoUREGVJQfa0KQ