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