linux中的 【信号】
信号的产生方式:
1、键盘输入
如 Ctrl+C 表示产生一个SIGINT信号
2、异常产生信号
如 程序执行到 2/0 这种情况、 管道的读段已经关闭而写端仍向管道中写入数据这种情况 等...
3、通过命令向指定进程发送信号
对信号的处理方式:
1、忽略
2、执行默认处理(通常为终止程序)
3、执行自定义动作 (信号的捕捉)
举个例子。
下面这段程序,从0开始,每隔一秒输出一个不断增长的数值
执行:
显然 它是一个死循环,将会一直执行下去
用信号中断它,可以直接用Ctrl + C
这样内核会对该进程发送一个 SIGINT 信号,结束当前进程:
此外,执行程序后 另外打开一个终端
(其实在执行命令的语句后面加上 & 就可以让当前进程在后台运行,也就不需要另外打开中断了,这里这样做是为了更直观一些)
先用 ps aux 命令查找到当前进程的 PID
然后用 killl -l 命令查看所有信号:
这里能够发现, Ctrl+C快捷键发送的 SIGINT 信号对应的2,
而我当前进程的PID 为:20262
那么,执行命令 : kill -2 20262 或者 kill -SIGINT 20262
发现死循环的进程终止了
接下来介绍几个信号集操作函数:
int sigemptyset(sigset_t *set);
int sigfillset(sigset_t *set);
int sigaddset(sigset_t *set, int signo);
int sigdelset(sigset_t *set, int signo);
int sigismember(const sigset_t *set, int signo);
int sigprocmask(int how, const sigset_t *set, sigset_t *oset);
int sigpending(sigset_t *set);
第一个函数:
sigemptyset 用来初始化 set 指向的 信号集,使所有信号对应的bit清零,表示该信号集不包含任何有效信号
成功返回0,出错返回 -1
第二个函数:
sigfillset 同样用来初始化,不过跟上面那个 sigemptyset 作用相反,它表示该信号集包括系统支持的所有信号
成功返回0,出错返回 -1
在使用 sigset_t 类型变量之前,必须调用上述两个初始化阐述中的任意一个
第三个函数:
sigaddset 的作用是向指定信号集 set 中添加 有效信号
成功返回0,出错返回 -1
第四个函数:
sigdelset 的作用是在指定信号集 set 中 删除有效信号
成功返回0,出错返回 -1
第五个函数:
sigismember 用来判断指定的信号是否在指定信号集中
若存在,返回1; 若不存在,返回0; 若出错,返回-1
第六个函数:
sigprocmask 可以读取或更改进程的信号屏蔽字
int sigprocmask(int how, const sigset_t *set, sigset_t *oset);
成功返回0,失败返回-1
其中第一个参数 how表示处理方式,有以下三种:
·SIG_BLOCK :
set包含了我们希望添加到当前信号屏蔽字的信号 //相当于mask = mask | set
·SIG_UNBLOCK :
set包含了我们希望从当前信号屏蔽字中解除阻塞的信号 //相当于 mask = mask | ~set
·SIG_SETMASK :
设置当前信号屏蔽字为set所指向的值 //相当于 mask = set
第七个函数:
sigpending 读取当前进程的未决信号集,通过set参数传出
成功返回0,失败返回-1
接下来,用上面介绍的7个函数,实现一段功能
这里我定义的 print_sig 函数的作用是 将指定信号集的所有信号的存在情况打印出来
main函数中, 第35行 将SIGINT信号添加进 s 信号集当中
然后37行的 sigprocmask 执行完毕后, SIGINT信号就相当于被阻塞了,同时,o保存的信号集就是之前的s(全 0)
while循环 每次读取并打印 当前的未决信号集 每循环5次就恢复一次阻塞,如果没有信号被阻塞,则打印“recover block” 并继续执行循环,
如果有信号被阻塞,那么在49行恢复阻塞完毕后,程序就会终止掉。
运行结果:
转载于:https://blog.51cto.com/zhweizhi/1825574