sigaction函数的功能是检查或修改指定信号相关联的处理动作,此函数取代UNIX早期版本使用的signal函数。
#include<signal.h> int sigaction(int signo, const struct sigaction *restrict act, struct sigaction *restrict oact); //若成功则返回0,出错则返回-1.
参数signo是要检测或修改其具体动作的信号编号。
若act指针非空,则要修改其动作;如果oact指针非空,则系统经由oact指针返回该信号的上一个动作,此函数使用如下结构:
struct sigction{ void (*sa_handler)(int); //信号捕捉函数的地址或者SIG_IGN,或者SIG_DFL sigset_t sa_mask; //在调用信号捕捉函数之前要屏蔽的信号,当从信号步骤函数返回时,将进程信号屏蔽字复原。 //操作系统建立的新信号屏蔽字包含正在被递送的信号。 int sa_flags; //信号选项 void (*sa_sigaction)(int, siginfo_t*, void*); //替代信号处理函数地址 }
act结构的sa_flags字段指定对信号处理的各个选项。
sa_sigaction字段是一个替代的信号处理程序,当在sigaction结构中使用了SA_SIGINFO标志时,使用该信号处理程序。
通常,按下列方式调用信号处理程序:
void handler(int signo);
但是,如果设置了SA_SIGINFO标志,那么按照下列方式调用信号处理程序:
void handler(int signo, siginfo_t *info, void *context);
siginfo_t结构包含了信号产生原因的有关信息,该结构的大致样式如下:
struct siginfo{ int sig_signo; //信号 int sig_errno; //如果不是0,就是errno.h中的errno值 int sig_code; //附加信息(取决于信号) pid_t si_pid; //发送信号的进程ID uid_t sig_uid; //发送信号的进程真实用户ID void *si_addr; //产生错误的地址 int si_status; //退出值或者信号值 long si_band; //SIGPOLL的band号 //可能还会有其他的值。 }
实践:
#include <stdio.h> #include <signal.h> static void sighandle(int signo){ printf("catch signo:%d\n",signo); } int main(void){ struct sigaction act; act.sa_handler = sighandle; sigemptyset(&act.sa_mask); act.sa_flags = 0; if(sigaction(SIGQUIT,&act,NULL) <0){ perror("sigaction"); return -1; } while(1){ pause(); } return 0; }
运行结果:
yan@yan-vm:~/apue$ ./a.out
^\catch signo:3
^\catch signo:3
^\catch signo:3
^\catch signo:3
^C