前期踩的坑 (spring boot 1.x)
1. 添加mavne依賴
1
2
3
4
5
|
<!-- springboot監控 --> <dependency> <groupid>org.springframework.boot</groupid> <artifactid>spring-boot-starter-actuator</artifactid> </dependency> |
2. 啟用shutdown
在配置文件里添加下面的配置
1
2
3
4
|
#啟用shutdown endpoint的http訪問 endpoints.shutdown.enabled= true #不需要驗證 endpoints.shutdown.sensitive= false |
啟動的時候可以看到下面的日志,就說明成功了
3. 優雅停機
發送post請求 http://localhost:8080/shutdown
如果響應碼是404 可以嘗試post http://localhost:8080/actuator/shutdown
spring boot 2.0
如果你使用的spring boot版本是2.x的就會發現,這些post請求都會出現404的結果。
下面是spring boot 2.0 優雅停機的實現方式。
1.修改application啟動類
tomcat容器
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
|
@springbootapplication public class shutdownapplication { public static void main(string[] args) { springapplication.run(shutdownapplication. class , args); } /** * 用于接受 shutdown 事件 */ @bean public gracefulshutdown gracefulshutdown() { return new gracefulshutdown(); } /** * 配置tomcat * * @return */ @bean public servletwebserverfactory servletcontainer() { tomcatservletwebserverfactory tomcat = new tomcatservletwebserverfactory(); tomcat.addconnectorcustomizers(gracefulshutdown()); return tomcat; } /** * 優雅關閉 spring boot。容器必須是 tomcat */ private class gracefulshutdown implements tomcatconnectorcustomizer, applicationlistener<contextclosedevent> { private final logger log = loggerfactory.getlogger(gracefulshutdown. class ); private volatile connector connector; private final int waittime = 10 ; @override public void customize(connector connector) { this .connector = connector; } @override public void onapplicationevent(contextclosedevent contextclosedevent) { this .connector.pause(); executor executor = this .connector.getprotocolhandler().getexecutor(); if (executor instanceof threadpoolexecutor) { try { threadpoolexecutor threadpoolexecutor = (threadpoolexecutor) executor; threadpoolexecutor.shutdown(); if (!threadpoolexecutor.awaittermination(waittime, timeunit.seconds)) { log.warn( "tomcat 進程在" + waittime + " 秒內無法結束,嘗試強制結束" ); } } catch (interruptedexception ex) { thread.currentthread().interrupt(); } } } } } |
undertow容器 (沒有使用過,不保證可用)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
|
@springbootapplication public class application { public static void main(string[] args) { springapplication.run(application. class , args); } /** * 優雅關閉 spring boot */ @component public class gracefulshutdown implements applicationlistener<contextclosedevent> { @autowired private gracefulshutdownwrapper gracefulshutdownwrapper; @autowired private servletwebserverapplicationcontext context; @override public void onapplicationevent(contextclosedevent contextclosedevent){ gracefulshutdownwrapper.getgracefulshutdownhandler().shutdown(); try { undertowservletwebserver webserver = (undertowservletwebserver)context.getwebserver(); field field = webserver.getclass().getdeclaredfield( "undertow" ); field.setaccessible( true ); undertow undertow = (undertow) field.get(webserver); list<undertow.listenerinfo> listenerinfo = undertow.getlistenerinfo(); undertow.listenerinfo listener = listenerinfo.get( 0 ); connectorstatistics connectorstatistics = listener.getconnectorstatistics(); while (connectorstatistics.getactiveconnections() > 0 ){} } catch (exception e){ // application shutdown } } } @component public class gracefulshutdownwrapper implements handlerwrapper{ private gracefulshutdownhandler gracefulshutdownhandler; @override public httphandler wrap(httphandler handler) { if (gracefulshutdownhandler == null ) { this .gracefulshutdownhandler = new gracefulshutdownhandler(handler); } return gracefulshutdownhandler; } public gracefulshutdownhandler getgracefulshutdownhandler() { return gracefulshutdownhandler; } } @component @allargsconstructor public class undertowextraconfiguration { private final gracefulshutdownwrapper gracefulshutdownwrapper; @bean public undertowservletwebserverfactory servletwebserverfactory() { undertowservletwebserverfactory factory = new undertowservletwebserverfactory(); factory.adddeploymentinfocustomizers(deploymentinfo -> deploymentinfo.addouterhandlerchainwrapper(gracefulshutdownwrapper)); factory.addbuildercustomizers(builder -> builder.setserveroption(undertowoptions.enable_statistics, true )); return factory; } } } |
2. 使用 kill命令殺死進程
使用下面的命令殺死進程。該命令是向 某個進程發送終止信號。
1
|
kill - 15 [pid] |
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。
原文鏈接:https://blog.csdn.net/u013451048/article/details/80194001