本文仅作为学习记录,非商业用途,侵删,如需转载需作者同意。
一、思考题回答
1、第二讲的问答:
Q:有这样一个C语言的init进程,它没有注册任何信号的 handler ,如果我们从Host Namespace 向它发送 SIGTERM ,会发生什么情况呢?
A:即使在宿主机上向容器1号进程发送 SIGTERM ,在1号进程没有注册 handler 的情况下,这个进程也不能被杀死。
原因:看内核中的代码 !(force && sig_kernel_only(sig)) ,虽然从不同的Namespace发送信号,但是 sig_kernel_only(sig) 对于SIGTERM 来说还是0,那么 !(1 && 0) = 1。
#define sig_kernel_only(sig) siginmask(sig, SIG_KERNEL_ONLY_MASK)
#define SIG_KERNEL_ONLY_MASK (\
rt_sigmask(SIGKILL) | rt_sigmask(SIGSTOP))
2、第三讲的问答:
Q:如果容器的init 进程创建了子进程B,B又创建了自己的子进程C,如果C运行完之后,退出成了僵尸进程,B进程还在运行,而容器的init进程还在不断的调用 waitpid() ,那么C这个僵死进程可以被回收吗?
A:2位同学的回答如下
1、这时C不会被回收,只有等到B也被杀死,C这个僵死进程也会变成孤儿进程,被init进程收养,进而被init的 wait 机制清理掉。
2、C 不会被回收,waitpid 仅等待直接 children 的状态变化。
为什么先进入到僵尸状态而不是直接消失,觉得是留个父进程一次机会,查看子进程的PID、终止状态(退出码,终止原因,比如是信号终止还是正常退出等)、资源使用信息。如果子进程直接消失,那么父进程没有机会掌握子进程的具体终止情况。
一般情况下,程序逻辑可能会依据子进程的终止情况,做出进一步的处理,比如Nginx Master进程获知worker进程异常退出,则再拉起一个worker进程。
3、第四讲的问答:
Q:下面的代码输出的结果是什么:
#include <stdio.h>
#include <signal.h>
typedef void (*sighandler_t)(int);
void sig_handler(int signo)
{
if (signo == SIGTERM) {
printf("received SIGTERM\n\n");
// Set SIGTERM handler to default
signal(SIGTERM, SIG_DFL);
}
}
int main(int argc, char *argv[])
{
//Ignore SIGTERM, and send SIGTERM
// to process itself.
signal(SIGTERM, SIG_IGN);
printf("Ignore SIGTERM\n\n");
kill(0, SIGTERM);
//Catch SIGERM, and send SIGTERM
// to process itself.
signal(SIGTERM, sig_handler);
printf("Catch SIGTERM\n");
kill(0, SIGTERM);
//Default SIGTERM. In sig_handler, it sets
//SIGTERM handler back to default one.
printf("Default SIGTERM\n");
kill(0, SIGTERM);
return 0;
}
A:可以参考用户 geek 2014 同学的答案。输出结果如下:
Ignore SIGTERM
Catch SIGTERM
received SIGTERM
Default SIGTERM
4、第五讲的问答:
感觉后续的意义不大,不写在这里了。
自己需要的时候,再看下。