容器中的应用优雅的退出
一、进程的退出 - Linux
1、kill 参数
1 | SIGHUP | 启动被终止的程序,可让该进程重新读取自己的配置文件,类似重新启动。 |
---|---|---|
2 | SIGINT | 相当于用键盘输入 [ctrl]-c 来中断一个程序的进行。 |
9 | SIGKILL | 代表强制中断一个程序的进行,如果该程序进行到一半,那么尚未完成的部分可能会有“半产品”产生,类似 vim会有 .filename.swp 保留下来。 |
15 | SIGTERM | 以正常的方式来终止该程序。由于是正常的终止,所以后续的动作会将他完成。不过,如果该程序已经发生问题,就是无法使用正常的方法终止时,输入这个 signal 也是没有用的。 |
19 | SIGSTOP | 相当于用键盘输入 [ctrl]-z 来暂停一个程序的进行。 |
2、信号
信号是一种进程间通信的形式。一个信号就是内核发送给进程的一个消息,告诉进程发生了某种事件。当一个信号被发送给一个进程后,进程会立即中断当前的执行流并开始执行信号的处理程序。如果没有为这个信号指定处理程序,就执行默认的处理程序。
进程需要为自己感兴趣的信号注册处理程序,比如为了能让程序优雅的退出(接到退出的请求后能够对资源进行清理)一般程序都会处理 SIGTERM 信号。与 SIGTERM 信号不同,SIGKILL 信号会粗暴的结束一个进程。因此我们的应用应该实现这样的目录:捕获并处理 SIGTERM 信号,从而优雅的退出程序。如果我们失败了,用户就只能通过 SIGKILL 信号这一终极手段了。除了 SIGTERM 和 SIGKILL ,还有像 SIGUSR1 这样的专门支持用户自定义行为的信号
二、容器中的信号
-
容器的 1 号进程为Dockerfile文件中启动命令进程。
-
只有容器中的 1 号进程能够收到信号。
-
如果需要使用容器动态化(即CMD命令为脚本启动),那么脚本进程就为1号进程 。
例如:/bin/bash /root/start.sh为1号进程,应用进程为其他级别。
-
如果既想容器动态化,又想优雅退出。那么可以由DockerDameon发送信号给1号进程,再有1号进程发送给应用进程,实现信号的传递。脚本如下:
#!/bin/bash
pid=0
term_handler() {
if [ $pid -ne 0 ]; then
kill -SIGTERM "$pid"
wait "$pid"
fi
exit 143;
}
trap 'kill ${!}; term_handler' SIGTERM
启动命令 &
pid="$!"
while true
do
tail -f /dev/null & wait ${!}
done