在日常生活中我们也经常面临许多的信号,手机通知、过红绿灯。。。这些信号在没有发生之前我们就知道这种信号产生我们需要干什么,那Linux里信号产生后,又怎么知道要做什么呢? -- 那当然是由程序员自己去设置啊
由于我们的用户空间代码执行到任何地方都有可能收到信号,所以信号相对于进程的控制流程来说是异步。
那这也意味着,当信号到来时,有可能我们的程序在做一件比较重要的事,这时就可能会暂缓处理这个信号
信号的处理方式: 默认 、忽略 、自定义
signal:
在我们平时运行一个进程时,我们想让它停下来,可以使用 CTRL + c (2),CTRL + \(3) 来停止它,为什么这两个操作就可以停止呢,当然是因为产生了信号,而这两个信号正对应停止操作
signal函数可以自定义一个信号的处理方式,那如果我们把 2号和3号信号变成了我们自定义的处理方式,那是不是我们的进程就停不下来了呢。。
其实不是的,9号信号依然可以杀死我们的进程,那如果我们自定义9号信号,是不是一个进程真的就杀不死了呢,hhh当然不是的,我们能想得到,那别人也能够想到,9号信号就是默认的处理方式,不可被更改
abort函数也是终止进程的函数,但是它不会有任何的清理工作,给自己发送终止信号6
实际执行信号的处理动作称为信号递达;
信号从产生到递达中间的状态成为信号未决;
信号被阻塞后会处于信号未决的的状态,直到进程解除对此信号的阻塞,才能完成信号递达。
信号被忽略不会有抵达动作。
在我们进程中存放着三张表:block、pending、handler。handler表里面是一个个函数指针。
每个信号都有两个标志位分别表示阻塞和未决。信号产生时,内核在进程控制块设置该信号的未决状态,直到信号递达才清除。
如果一个信号在阻塞时,同一信号产生多次,Linux下只会计一次,也就是说多余的信号不会递达
sigset_t:
未决和阻塞可以用相同的数据类型sigset_t来存储,sigset_t称为信号集,这个类型可以表示每个信号的有效和无效的状态,用一个bit表示
信号集函数:
第一个函数是用来将信号集置0;
第二个函数是用来将信号集置1;
第三个函数是将某个信号加入信号集;
第四个函数是删除操作;
第五个函数是查看某一信号是否在一个信号集中;
这个函数是对block表进行修改的函数,可以添加、修改、自己想要阻塞的信号以及删除。
成功:返回0;失败:返回-1,设置errno
参数how:
- SIG_BLOCK:表示添加阻塞的信号
- SIG_UNBLOCK: 表示去掉信号的阻塞
- SIG_SETBLOCK: 用参数中的set覆盖掉原始的
参数oldset:
这个参数是一个输出型参数,输出修改之前的block表。
这个是一个获取当前pending表的函数,就是查看现在那些信号正处于未决状态。
参数set:
输出型参数
这个函数会是最常用的函数之一,自定义信号的处理方式,防止进程的意外死亡。
参数act:
一个数组指针;
act.sa_handler:自定义函数
act.sa_mask:说明额外需要屏蔽信号
当我们正在处理一个信号时,同一信号到来,会被阻塞直到信号处理完成
act.sa_flags:位掩码,指定信号处理过程的各种选项
问题:进程什么时候执行信号的处理动作?
答:当前进程从内核态切回用户态的时候,会进行信号的检测与执行。