信号的产生方式:

1、键盘输入

 如 Ctrl+C 表示产生一个SIGINT信号

 

2、异常产生信号

 如 程序执行到 2/0 这种情况、 管道的读段已经关闭而写端仍向管道中写入数据这种情况 等...

 

3、通过命令向指定进程发送信号

 

 

对信号的处理方式:

1、忽略

2、执行默认处理(通常为终止程序)

3、执行自定义动作 (信号的捕捉)

 

举个例子。

下面这段程序,从0开始,每隔一秒输出一个不断增长的数值

linux中的 【信号】

 

 执行:

linux中的 【信号】

 

显然 它是一个死循环,将会一直执行下去

用信号中断它,可以直接用Ctrl + C

这样内核会对该进程发送一个 SIGINT 信号,结束当前进程:

linux中的 【信号】

 

 

此外,执行程序后 另外打开一个终端 

 (其实在执行命令的语句后面加上 & 就可以让当前进程在后台运行,也就不需要另外打开中断了,这里这样做是为了更直观一些)

先用 ps aux 命令查找到当前进程的 PID

然后用 killl -l 命令查看所有信号:

linux中的 【信号】

 

 这里能够发现, 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个函数,实现一段功能

 

 

 

linux中的 【信号】

这里我定义的 print_sig 函数的作用是 将指定信号集的所有信号的存在情况打印出来

 

main函数中, 第35行 将SIGINT信号添加进 s 信号集当中

然后37行的 sigprocmask 执行完毕后,  SIGINT信号就相当于被阻塞了,同时,o保存的信号集就是之前的s(全 0)

while循环 每次读取并打印 当前的未决信号集  每循环5次就恢复一次阻塞,如果没有信号被阻塞,则打印“recover block” 并继续执行循环,

如果有信号被阻塞,那么在49行恢复阻塞完毕后,程序就会终止掉。

 

运行结果:

linux中的 【信号】