sigaction函数
该函数补充signal函数的缺陷,对signal的扩展,signal不知道谁发的信号
|#include <signal.h>
int sigaction(
int signum, //信号码
const struct sigaction* act, //信号处理方式
struct sigaction* oldact //原信号处理方式,可为NULL
);
struct sigaction{
void (sa_handler)(int); //信号处理函数指针1
void (sa_sigaction)(int, siginfo_t, void); //信号处理函数指针2
sigset_t sa_mask; //信号掩码 (信号处理函数执行中才屏蔽)
int sa_flags; //信号处理标志
void (*sa_restorer)(void); //保留,NULL
}
成功返回0 ,失败返回-1
typedef struct siginfo{
pid_t si_pid; //发送信号的pid
sigval_t si_value; //信号附加值
......
} siginfo_t;
typedef union sigval{ //联合
int sigval_int; //或者传整数
void* sival_ptr; //或者传结构体
} sigval_t;
1、缺省情况下,在信号处理函数的执行过程中,
会自动屏蔽这个正在被处理的信号,
而对其它信号则不屏蔽。
通过sigaction::sa_mask成员可以人为指定,
在信号处理函数的执行过程中,
需要加入进程信号掩码中的信号,
并在信号处理函数执行完后,
自动解除对这些信号的屏蔽。
2、sigaction::sa_flags可为以下值的位或。
SA_ONESHOT/SA_RESETHAND 两个一样
- 执行完一次信号处理函数后,即将对此信号的处理恢复为默认处理的方式
这也是老版本的signal函数的缺省行为
SA_NODEFER/SA_NOMASK 允许重入,用的多
-在信号处理函数的执行过程中,不屏蔽这个正在处理的信号
SA_NOCLDSTOP
-若signum函数取SIGCHLD,则当子进程暂停时,不通知父进程
SA_RESTART
-系统调用一旦被signum参数所表示的信号中断,会自动重启
SA_SIGINFO 用的多
-使用信号指针2,通过这个函数的第二个参数提供更多的信息
|#include <stdio.h>
|#include <stdlib.h>
|#include <signal.h>
|#include <unistd.h>
void sigint1(int signum){
printf("收到SIGINT信号!睡眠中...\n");
sleep(5);
printf("睡醒了。\n");
}
void sigint2(int signum, siginfo_t* si,void* pv){
printf("收到%u进程的SIGINT信号!睡眠中...\n",si->si_pid);
}
int main(void){
printf("按<Ctrl+c>和<Ctrl+\>看效果\n");
struct sigaction act = {};
printf("睡眠中只屏蔽SIGINT,不屏蔽SIGQUIT.\n");
act.sa_handler = sigint1;
if(sigaction(SIGINT, &act, NULL) == -1){
perror("sigaction");
return -1;
}
for(;;);
}
- 以上为老版本信息。以下为新版本功能.
|#include <stdio.h>
|#include <stdlib.h>
|#include <signal.h>
|#include <unistd.h>
void sigint1(int signum){
printf("收到SIGINT信号!睡眠中...\n");
sleep(5);
printf("睡醒了。\n");
}
void sigint2(int signum, siginfo_t* si,void* pv){
printf("收到%u进程的SIGINT信号!睡眠中...\n",si->si_pid);
}
int main(void){
printf("按<Ctrl+c>和<Ctrl+\>看效果\n");
struct sigaction act = {};
//printf("睡眠中只屏蔽SIGINT,不屏蔽SIGQUIT.\n");
//act.sa_handler = sigint1;
printf("睡眠中既屏蔽SIGINT,也屏蔽SIGQUIT.\n"); //睡醒之后才退出
act.sa_handler = sigint1;
sigemptyset(&act.sa_mask);
sigaddset (act.sa_mask, SIGQUIT); //屏蔽SIGQUIT信号
act.sa_flags = SA_NOMASK; // 达到不屏蔽SIGINT的效果。函数可重入
if(sigaction(SIGINT, &act, NULL) == -1){
perror("sigaction");
return -1;
}
for(;;);
}
- 如果需要附带更多的信息
|#include <stdio.h>
|#include <stdlib.h>
|#include <signal.h>
|#include <unistd.h>
void sigint1(int signum){
printf("收到SIGINT信号!睡眠中...\n");
sleep(5);
printf("睡醒了。\n");
}
void sigint2(int signum, siginfo_t* si,void* pv){
printf("收到%u进程的SIGINT信号!睡眠中...\n",si->si_pid);
}
int main(void){
printf("按<Ctrl+c>和<Ctrl+\>看效果\n");
struct sigaction act = {};
//printf("睡眠中只屏蔽SIGINT,不屏蔽SIGQUIT.\n");
//act.sa_handler = sigint1;
printf("睡眠中既屏蔽SIGINT,也屏蔽SIGQUIT.\n"); //睡醒之后才退出
act.sa_sigaction = sigint2;
//sigemptyset(&act.sa_mask);
//sigaddset (act.sa_mask, SIGQUIT); //屏蔽SIGQUIT信号
act.sa_flags = SA_NOMASK | SA_SIGINFO; // 达到不屏蔽SIGINT的效果。函数可重入..使用第二个参数,发送更多的信息
if(sigaction(SIGINT, &act, NULL) == -1){
perror("sigaction");
return -1;
}
for(;;);
}