函数函数sigaction、signal

时间:2023-03-09 09:35:44
函数函数sigaction、signal

函数函数sigaction

1. 函数sigaction原型:

int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);

分析:

  • 参数 signum :要捕获的信号。
  • 参数act:truct sigaction 结构体,后面具体讲解传入参数,新的处理方式
  • 参数oldact:返回旧的 struct sigaction 结构体,传出参数,旧的处理方式

1.2 struct sigaction 结构体

 struct sigaction
{
void(*sa_handler)(int);
void(*sa_sigaction)(int, siginfo_t *, void *);
sigset_t sa_mask;
int sa_flags;
void(*sa_restorer)(void);
};

分析:

  • sa_handler : 不带附加参数的信号处理函数指针
  • sa_sigaction: 带有附加参数的信号处理函数指针(两个信号处理函数指针只能二选一)
  • sa_mask: 在执行信号处理函数时,应该屏蔽掉哪些信号
  • sa_flags: 用于控制信号行为,它的值可以是下面选项的组合。
  • SA_SIGINFO:如果指定该选项,则向信号处理函数传递参数(这时应该使用 sa_sigaction 成员而不是 sa_handler).
  • sa_restorer:该成员在早期是用来清理函数栈的,如今已被废弃不用。

注意:sa_flags 的选项比较多,大部分可又自己做实验验证,有些是需要额外的知识,比如 SA_ONESTACK 和 SA_RESTART,这些放到后面讲解。本节示例中,只需要把 sa_flags 设置为 0 即可。

信号捕捉特性:

  • 进程正常运行时,默认PCB有一个信号屏蔽字,假定为☆,它决定了进程自动屏蔽哪些信号,当注册了某个信号步捕捉函数,捕捉到该信号以后,要调用该函数,而该函数有可能执行很长时间,在这期间所屏蔽的信号不由☆指定,而由sa_mask来指定,调用完信号处理函数,再恢复为☆。
  • xxx信号捕捉函数执行期间,XXX信号自动屏蔽。
  • 阻塞的常规信号不支持排队,产生多次只记录一次(后32个实时信号支持排队)

1. 测试代码:

 #include<stdio.h>
#include<signal.h>
#include<stdlib.h>
#include<unistd.h> void docatch(int signo)
{
printf("%d signal is catch\n", signo);
} int main()
{
int ret;
struct sigaction act;
act.sa_handler = docatch;
sigemptyset(&act.sa_mask);
sigaddset(&act.sa_mask, SIGQUIT);
act.sa_flags = ; //默认属性:信号捕捉函数执行期间,自动屏蔽本信号
ret = sigaction(SIGINT, &act, NULL);
if(ret < )
{
perror("sigaction error");
exit();
}
while()
sleep();
return ;
}

输出结果:

函数函数sigaction、signal

2. 测试代码:

 #include<stdio.h>
#include<signal.h>
#include<stdlib.h>
#include<unistd.h> void docatch(int signo)
{
printf("%d signal is catch\n", signo);
sleep();
printf("--------------finish-\n");
} int main()
{
int ret;
struct sigaction act;
act.sa_handler = docatch;
sigemptyset(&act.sa_mask);
sigaddset(&act.sa_mask, SIGQUIT);
act.sa_flags = ;
ret = sigaction(SIGINT, &act, NULL);
if (ret < )
{
perror("sigaction error");
exit();
}
while ()
sleep();
return ;
}

输出结果:

函数函数sigaction、signal

3. 测试代码:

 #include <unistd.h>
#include <signal.h>
#include <stdio.h> void printsigset(const sigset_t *set)
{
for (int i = ; i <= ; i++)
{
if (i == ) putchar(' ');
if (sigismember(set, i) == )
putchar('');
else
putchar('');
}
puts("");
} void handler(int sig)
{
if (sig == SIGTSTP)
printf("hello SIGTSTP\n");
if (sig == SIGINT)
printf("hello SIGINT\n");
sleep();
sigset_t st;
sigpending(&st);
printsigset(&st);
} int main()
{
printf("I'm %d\n", getpid());
struct sigaction act, oldact;
act.sa_handler = handler; // 设置普通信号处理函数
sigemptyset(&act.sa_mask); // 向 sa_mask 中添加 SIGINT
sigaddset(&act.sa_mask, SIGINT);
act.sa_flags = ; // 先置 0 sigaction(SIGTSTP, &act, &oldact);
sigaction(SIGINT, &act, &oldact); while ()
{
write(STDOUT_FILENO, ".", );
pause();
}
return ;
}

输出结果:

函数函数sigaction、signal

分析:

  • 当程序运行的时候,Ctrl C 进入 handler,然后立即 Ctrl Z 发现 handler 还未执行完就被 SIGTSTP 打断.
  • 当程序运行的时候,Ctrl Z 进入 handler,然后立即 Ctrl C 发现并不会被 SIGINT 打断,这是因为该 handler 注册的时候被设置了 SA_MASK = SIGINT。最后 handler 结束的时候打印了未决信号集,发现里头有 SIGINT。所以 handler 结束后,又去继续对 SIGINT 进行处理。