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