Java中可以添加ShutdownHook監聽關閉事件,包括kill -15, control+c,terminal等信號。kill -9則接收不到。
Runtime.getRuntime().addShutdownHook(new ShutdownThread());
如果java運行在容器中,stop docker容器時,容器內的java進程不一定能接收到kill事件。
原因主要是:dockerfile中 ENTRYPOINT 我們經常運行的腳本,在腳本中設置好java啟動參數,再java -jar啟動進程。所以docker 容器內的1號進程是shell腳本,而shell(sh)不會把自己收的signals傳遞給子進程,所以容器shell的子進程都接收不到kill 信號。
exec表示取代當前shell,關閉當前的shell process,換到exec后面的命令繼續執行,不產生新進程。shell中exec命令行之后的腳本也不會再執行。
解決方式:
1、使用supervisor管理java進程,shell中用 exec java -jar啟動。
2、從ENTRYPOINT 到java進程,都用exec執行
bash 傳遞kill signal,可以參考:https://unix.stackexchange.com/questions/146756/forward-sigterm-to-child-in-bash
參考:
Gracefully Shutting Down Java in Containers:https://dzone.com/articles/gracefully-shutting-down-java-in-containers
SIGTERM not received by java process using 'docker stop' and the official java image:https://stackoverflow.com/questions/31836498/sigterm-not-received-by-java-process-using-docker-stop-and-the-official-java-i