一、信号捕捉
信号捕捉针对于自定义的函数处理信号方式。信号递达调用这个函数称为捕捉信号。发生信号并不是立即处理的,而是找合适的机会,这个机会就是从内核态切换到用户态的时候处理信号。
用户态切换到内核态的方式:(1)系统调用(2)程序异常(3)软件中断
信号捕捉流程顺序见下图:
二、pause函数
调用pause函数回事进程挂起,直到有信号递达。如果信号的处理动作是终止进程,则进程终止,pause函数没有机会返回。
如果信号的处理动作是忽略的,则进程继续处于挂起状态,pause不返回。如果信号的处理动作为捕捉,则调用处理函数之后pause函数返回-1,error设置为EINTR,所以pause只有出错返回值,没有成功返回值和我们之前讲过的程序替换函数一样。
三、SIGCHLD信号
我们之前涉及到的僵尸进程,是由于子进程先于父进程退出,父进程没有关心子进程的退出状态导致僵尸进程。其实子进程退出会给父进程发送一个信号就是SIGCHLD。之前避免僵尸进程一种方法就是阻塞式的等待,父进程就不能处理自己的工作。第二种就是waitpid轮询等待的方式,但是这种父进程一边要处理自己的工作,另外一边还要轮询查询子进程是否结束,这样程序实现起来比较复杂。
我们可以使用该信号来处理僵尸进程,通过对该信号的自定义处理函数,然后父进程只管处理自己的事情,如果该信号到来父进程去处理,处理完毕之后接着做父进程的事情。
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<signal.h>
#include<sys/wait.h>
void sigcb(int signo)
{
while(waitpid(-1,0,WNOHANG)>0)
{
printf("signo:%d\n",signo);
}
}
int main()
{
signal(SIGCHLD,sigcb);
int pid = fork();
if(pid<0)
{
perror("fork error");
exit(1);
}
else if(pid == 0)
{
sleep(5);
exit(1);
}
else
{
while(1)
{
printf("waiting!!!\n");
sleep(1);
}
}
return 0;
}