背景知识
signal.h是C标准函数库中的信号处理部分, 定义了程序执行时如何处理不同的信号。信号用作进程间通信, 报告异常行为(如除零)、用户的一些按键组合(如同时按下Ctrl与C键,产生信号SIGINT)。信号是程序执行过程中发生的异常事件,同步信号的产生 是因为程序自身的某些动作,例如除零或不正当的访问存储器,异步信号是由程序外部的行为引起的,比如有人敲击了提示键或者另外一个程序(异步地执行)给你 的程序发信号,都会引发一个异步信号。程序不能屏蔽的信号要求立即得到处理。如果不对发生的信号指定一种处理方法,那么它就会被作为一种致命错误对待,然 后程序就会以失败的状态终止执行
头文件<singal.h>定义了一个无穷信号集的各种编码值:
#define _NSIG 16 //能够使用的信号的数量 #define SIGHUP 1 // 挂起 #define SIGINT 2 // 中断 (DEL) #define SIGQUIT 3 // 退出 (ASCII FS) #define SIGILL 4 // 非法指令 #define SIGTRAP 5 // 跟踪中断 (捕捉的时候没有重置) #define SIGABRT 6 // IOT 指令 #define SIGIOT 6 // 终止使用PDP-11 #define SIGUNUSED 7 // 备用代码 #define SIGFPE 8 // 浮点型异常 #define SIGKILL 9 // kill (不能捕获或忽略) #define SIGUSR1 10 // 用户定义信号 # 1 #define SIGSEGV 11 // 内存段异常 #define SIGUSR2 12 // 用户定义信号 # 2 #define SIGPIPE 13 // 在管道上写操作没有其他的读操作 #define SIGALRM 14 // 警告 #define SIGTERM 15 // kill后获得软件终止信号 #define SIGEMT 7 // 旧式 #define SIGBUS 10 // 旧式 /* POSIX 要求定义下面的信号, 即使它们不被支持. 这里给出这些定义, 但是它们不被支持. */ #define SIGCHLD 17 // 子进程终止或停止 #define SIGCONT 18 // 如果停止则继续 #define SIGSTOP 19 // 停止信号 #define SIGTSTP 20 // 交互式停止信号 #define SIGTTIN 21 // 后台进程试图读 #define SIGTTOU 22 // 后台进程试图写
两个函数:
raise函数,报告一个同步信号
int raise (int _sig);
signal函数,可以指定一种信号的处理方法
__sighandler_t signal (int _sig, __sighandler_t _func) ;
可以通过一下三种方式处理信号:
-
默认处理是终止程序,就像上面所描述的那样
-
信号忽略是直接把它丢弃
-
信号处理是把控制权移给一个指定的函数
C标准的内容
头文件 <signal.h>为处理各种各样的信号声明了一个类型和两个函数,并定义了几个宏
定义的类型是:sig_atomic_t
typedef int sig_atomic_t;
该类型是整数类型,声明为这种类型的对象可以作为一个原子实体被访问,即使有异步中断发生的时候也是如此。
定义的宏有: SIG_DFL、SIG_ERR、SIG_IGN
#define SIG_ERR ((__sighandler_t) -1) // 出错返回值 #define SIG_DFL ((__sighandler_t) 0) // 默认信号处理 #define SIG_IGN ((__sighandler_t) 1) // 忽略信号
它们展开为具有不同值的常量表达式,这些表达式的类型和函数signal的第二个参数及返回值的类型兼容,并且它们的值与所有声明的函数的地址都不相等。
指定信号处理
函数signal:
typedef void (*__sighandler_t) (int); __sighandler_t signal(int _sig, __sighandler_t _func);
或声明成:
void (*signal(int sig, void (*func) (int)))(int);
signal函数通过3种方式来保证接收到的信号编号sig被依次处理,如果func的值是SIG_DFL,那么会使用默认的信号处理方式;如果 func的值是SIG_IGN,那么这个信号就会被忽略;否则,func就指向一个函数,当这个信号发生时,就调用这个函数,这样的函数就被称为信号处理 程序。
函数raise:
int raise(int sig);
函数raise把信号sig送给正在执行的程序
返回值:执行成功返回0,否则返回非0
<singal.h>的使用
从本质上说,信号处理是不可移植的。只有当必须指定一个已知的操作系统集的信号处理时,才能使用<signal.h>中声明的函数,不用费尽心思把代码变成通用的,这是不可能的
如果对一个信号的默认处理是可行的,那么就选择这种处理方法,使用自己的信号处理程序会降低可移植性并且可能导致程序对信号的误处理增多,如果必须提供某个信号的处理程序,对其进行分类如下:
不可能返回值的信号处理程序,e.g:SIGFPE和SIGABRT
必须返回值的信号处理程序,e.g:SIGINT
不能返回值的信号处理程序最后会调用abort、exit或者longjmp来结束。当然,SIGABRT的处理程序不能调用abort来结束。处 理程序不应该调用singal使自己重建。如果程序没有终止,就把剩下的工作给其他的函数。如果信号是异步的,要注意程序所有的输入输出,因为这样的操作 也会中断库的执行。
必须返回的信号处理程序用一个return语句结束,如果它想让自己复位,那么应该在入口处立即返回。如果信号是异步的,就在一个 sig_atomic_t类型的volatile数据对象中存储一个非零值。不要执行其他任何对正在执行的程序有副作用的动作,例如输入输出和访问其他的 数据对象
<singal.h>的实现
下面是一个异步信号处理程序的例子:
#include<signal.h> static sig_atomic_t intflag = 0; static void field_int(int sig) { signal(SIGINT, &field_int); intflag = 1; return; }
这个程序调用signal(SIGINT, &field_int)来创建处理程序。它会不时地执行下面的代码来检查是否发生了异步交互式提示中断:
if(intflag) { //执行中断 intflag = 0; ... }
raise.c
#include<stdio.h> #include<signal.h> #include<stdlib.h> _sigfun *_sigtable[_NSIG] = {0}; int raise(int sig) { _sigfun *s; if(sig <= 0 || _NSIG <= sig) return -1; if((s = _sigtable[sig]) != SIG_IGN && s!= SIG_DFL) { _sigtable[sig] = SIG_DFL; (*s)(sig); } else if (s == SIG_DFL) { char ac[10], *p; switch(sig) { case SIGABRT: p = "abort"; break; case SIGFPE: p = "arithmetic error"; break; case SIGILL: p = "invalid executable code"; break; case SIGINT: p = "interruption"; break; case SIGSEGV: p = "invalid storage access"; break; case SIGTERM: p = "termination request"; break; default: *(p = &ac[sizeof(ac) - 1]) = '\0'; do *--p = sig % 10 + '0'; while((sig /= 10) != 0); fputs("signal #", stderr); break; } fputs(p, strerr); fputs(" -- terminating\n", stderr); exit(EXIT_FAILURE); } return 0; }
signal.c
#include<signal.h> _sigfun *_sigtable[_NSIG]; _sigfun *signal(int sig, _sigfun *fun) { _sigfun *s; if(sig <= 0 || _NSIG <= sig || fun == SIG_ERR) return SIG_ERR; s = _sigtable[sig]; _sigtable[sig] = fun; return s; }