信号
- 信号是进程在运行过程中,由自身产生或由进程外部发过来的消息(事件)。
- 信号是硬件中断的软件模拟(软中断)。
- 每个信号用一个整型常量宏表示,以SIG开头,比如SIGCHLD、SIGINT等,它们在系统头文件
中定义,也可以通过在shell下键入* kill –l查看信号列表,或者键入man 7 signal**查看更详细的说明
信号的生成
- 信号的生成来自内核,让内核生成信号的请求来自3个地方
- 用户
- 用户能够通过输入CTRL+c、Ctrl+\,或者是终端驱动程序分配给信号控制字符的其他任何键来请求内核产生信号
- 内核
- 当进程执行出错时,内核会给进程发送一个信号,例如非法段存取(内存访问违规)、浮点数溢出等
- 进程
- 一个进程可以通过系统调用kill给另一个进程发送信号,一个进程可以通过信号和另外一个进程进行通信
- 用户
信号类别
- 由进程的某个操作产生的信号称为同步信号(synchronous signals)
- 由像用户击键这样的进程外部事件产生的信号叫做异步信号(asynchronous signals)
信号的处理方式
进程接收到信号以后,可以有如下3种选择进行处理
- 接收默认处理:接收默认处理的进程通常会导致进程本身消亡。
- 例如连接到终端的进程,用户按下CTRL+c,将导致内核向进程发送一个SIGINT的信号,进程如果不对该信号做特殊的处理,系统将采用默认的方式处理该信号,即终止进程的执行; signal(SIGINT,SIG_DFL)
- 忽略信号:进程可以通过代码,显示地忽略某个信号的处理
- 例如:signal(SIGINT,SIG_IGN);但是某些信号是不能被忽略的
- 捕捉信号并处理:进程可以事先注册信号处理函数,当接收到信号时,由信号处理函数自动捕捉并且处理信号
有两个信号既不能被忽略也不能被捕捉,它们是SIGKILL和SIGSTOP。即进程接收到这两个信号后,只能接受系统的默认处理,即终止进程
signal信号处理机制
1 2 3 4 5 |
#include <signal.h> typedef void (*sighandler_t)(int); //函数指针 sighandler_t signal(int signum, sighandler_t handler); |
- signal的第1个参数signum表示要捕捉的信号
- 第2个参数是个函数指针,表示要对该信号进行捕捉的函数,该参数也可以是SIG_DFL(表示交由系统缺省处理,相当于白注册了)或SIG_IGN(表示忽略掉该信号而不做任何处理)。
- signal如果调用成功,返回以前该信号的处理函数的地址,否则返回SIG_ERR。
- sighandler_t是信号捕捉函数,由signal函数注册,注册以后,在整个进程运行过程中均有效,并且对不同的信号可以注册同一个信号捕捉函数。该函数只有一个整型参数,表示信号值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
//捕捉终端CTRL+c产生的SIGINT信号 #include <unistd.h> #include <stdio.h> #include <sys/wait.h> #include <sys/types.h> #include <signal.h> void SignHandler(int iSignNo) { printf("Capture sign no:%d\n",iSignNo); } int main() { signal(SIGINT,SignHandler); while(1) sleep(1); return 0; } |
该程序运行起来以后,通过按CTRL+c将不再终止程序的运行(或者另开一个终端,然后发送消息:“kill –s 2 进程号”或者“kill -2 进程号”,可以实现Ctrl + c同样的效果。因为CTRL+c产生的SIGINT信号已经由进程中注册的SignHandler函数捕捉了。该程序可以通过Ctrl+\终止,因为组合键Ctrl+\能够产生SIGQUIT信号,而该信号的捕捉函数尚未在程序中注册。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
//忽略掉终端CTRL+c产生的SIGINT信号 #include <unistd.h> #include <stdio.h> #include <sys/wait.h> #include <sys/types.h> #include <signal.h> int main() { signal(SIGINT,SIG_IGN); while(1) sleep(1); return 0; } |
该程序运行起来以后,将CTRL+C产生的SIGINT信号忽略掉了,所以CTRL+C将不再能使该进程终止,要终止该进程,可以向进程发送SIGQUIT信号,即组合键CTRL+\。或者另外开一个端口,然后执行ps –aux查看进程,发现该进程号之后用kill -9 进程号 杀掉该进程
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
// 接受信号的默认处理,接受默认处理就相当于没有写信号处理程序 #include <unistd.h> #include <stdio.h> #include <sys/wait.h> #include <sys/types.h> #include <signal.h> int main() { signal(SIGINT,SIG_DFL); while(1) sleep(1); return 0; } |
sigaction信号处理机制
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
#include <signal.h> int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact); struct sigaction { void (*sa_handler)(int); //老类型的信号处理函数指针 void (*sa_sigaction)(int, siginfo_t *, void *);//新类型的信号处理函数指针 sigset_t sa_mask; //将要被阻塞的信号集合 int sa_flags; //信号处理方式掩码 (SA_SIGINFO ) void (*sa_restorer)(void); //保留,不要使用 }; //字段sa_handler是一个函数指针,用于指向原型为void handler(int)的信号处理函数地址, 即老类型 的信号处理函数(如果用这个再将sa_flags = 0,就等同于signal()函数) //字段sa_sigaction也是一个函数指针,用于指向原型为 //原型为: //void handler(int iSignNum, siginfo_t *pSignInfo, void *pReserved); //iSignNum:传入的信号 //pSignInfo:与该信号相关的一些信息,它是个结构体 //pReserved:保留,现没用,通常为NULL |
- sigaction也用于注册一个信号处理函数
- 参数signum为需要捕捉的信号
- 参数act是一个结构体,里面包含信号处理函数地址、处理方式等信息
- 参数oldact是一个传出参数,sigaction函数调用成功后,oldact里面包含以前对signum的处理方式的信息,通常为NULL
- 如果函数调用成功,将返回0,否则返回-1
- 字段sa_handler和sa_sigaction只应该有一个生效,如果想采用老的信号处理机制,就应该让sa_handler指向正确的信号处理函数,并且让字段sa_flags为0;否则应该让sa_sigaction指向正确的信号处理函数,并且让字段sa_flags包含SA_SIGINFO选项
- 字段sa_mask是一个包含信号集合的结构体,该结构体内的信号表示在进行信号处理时,将要被阻塞的信号。针对sigset_t结构体,有一组专门的函数对它进行处理
1 2 3 4 5 6 7 8 |
#include <signal.h> int sigemptyset(sigset_t *set); //清空信号集合set int sigfillset(sigset_t *set); //将所有信号填充进set中 int sigaddset(sigset_t *set, int signum); //往set中添加信号signum int sigdelset(sigset_t *set, int signum); //从set中移除信号signum int sigismember(const sigset_t *set, int signum); //判断signum是否包含在set中(是:返回1,否:0) int sigpending(sigset_t *set); //将被阻塞的信号集合由参数set指针返回 //其中,对于函数sigismember而言,如果signum在set集中,则返回1;不在,则返回0;出错时返回-1.其他的函数都是成功返回0,失败返回-1. |
- 字段sa_flags是一组掩码的合成值,指示信号处理时所应该采取的一些行为,各掩码的含义为
掩码 | 描述 |
SA_RESETHAND | 处理完毕要捕捉的信号后,将自动撤消信号处理函数的注册,即必须再重新注册信号处理函数,才能继续处理接下来产生的信号。该选项不符合一般的信号处理流程,现已经被废弃。 |
SA_NODEFER | 在处理信号时,如果又发生了其它的信号,则立即进入其它信号的处理,等其它信号处理完毕后,再继续处理当前的信号,即递规地处理。如果**sa_flags包含了该掩码,则结构体sigaction的sa_mask**将无效!(不常用) |
SA_RESTART | 如果在发生信号时,程序正阻塞在某个系统调用,例如调用read()函数,则在处理完毕信号后,接着从阻塞的系统返回。如果不指定该参数,中断处理完毕之后,read函数读取失败。 |
SA_SIGINFO | 指示结构体的信号处理函数指针是哪个有效,如果sa_flags**包含该掩码,则sa_sigaction指针有效,否则是sa_handler指针有效**。(常用) |
本文为原创文章,版权归Aet所有,欢迎分享本文,转载请保留出处!
你可能也喜欢
- ♥ Linux 高性能服务器编程:IP协议09/04
- ♥ Linux 高性能服务器编程:高级I/O函数11/28
- ♥ 【Manjaro】Vmware分辨率不能修改03/22
- ♥ 包管理器:各平台安装卸载相关记述09/17
- ♥ Linux_ 命令大全 网络通讯03/16
- ♥ X86_64汇编学习记述三08/08