一个进程收到信号:执行默认动作、忽略信号、执行自定义动作。
【注意】:SIGKILL 和 SIGSTOP 不能更改信号的处理方式,因为它们向用户提供了一种使进程终止的可靠方法
1、signal函数
#include<signal.h>
typedef void(*sighandler_t)(int);
sighandler_t signal(int signum,sighandler_t handler);
功能:
注册信号处理函数(不可用于 SIGKILL、SIGSTOP 信号),即确定收到信号后处理函数的入口地址。
此函数不会阻塞。
参数:
signum:信号的编号,这里可以填数字编号,也可以填信号的宏定义,可以通过命令kill - l("l" 为字
母)进行相应查看。
handler : 取值有 3 种情况:
SIG_IGN:忽略该信号SIG_DFL:执行系统默认动作信号处理函数名:自定义信号处理函数,如:func
回调函数的定义如下:
void func(int signo)
{
// signo 为触发的信号,为 signal() 第一个参数的值
}
返回值:
成功:第一次返回NULL,下一次返回此信号上一次注册的信号处理函数的地址。如果需要使
用此返回值,必须在前面先声明此函数指针的类型。
失败:返回 SIG_ERR
2、sigaction函数
#include<signal.h>
int sigaction(int signum,const struct sigaction *act,struct sigaction*oldact);
功能:
检查或修改指定信号的设置(或同时执行这两种操作)。
参数:
signum:要操作的信号。
act: 要设置的对信号的新处理方式(传入参数)。
oldact:原来对信号的处理方式(传出参数)。
如果 act 指针非空,则要改变指定信号的处理方式(设置),如果 oldact 指针非空,则系统将此前指定
信号的处理方式存入 oldact。
返回值:
成功:0
失败:-1
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); //已弃用
};
1) sahandler、sasigaction:信号处理函数指针,和 signal() 里的函数指针用法一样,应根据情况给 sasigaction、sahandler 两者之一赋值,其取值如下:
a)SIGIGN:忽略该信号 b) SIGDFL:执行系统默认动作 c) 处理函数名:自定义信号处理函数
2) samask:信号阻塞集,在信号处理函数执行过程中,临时屏蔽指定的信号。
3) saflags:用于指定信号处理的行为,通常设置为 0,表使用默认属性。它可以是一下值的“按位或”组合:
Ø SARESTART:使被信号打断的系统调用自动重新发起(已经废弃)
Ø SANOCLDSTOP:使父进程在它的子进程暂停或继续运行时不会收到 SIGCHLD 信号。
Ø SANOCLDWAIT:使父进程在它的子进程退出时不会收到 SIGCHLD 信号,这时子进程如果退
出也不会成为僵尸进程。
ØSANODEFER:使对信号的屏蔽无效,即在信号处理函数执行期间仍能发出这个信号。
Ø SARESETHAND:信号处理之后重新设置为默认的处理方式。
ØSASIGINFO:使用 sasigaction 成员而不是 sahandler 作为信号处理函数。
信号处理函数:
void(*sa_sigaction)(int signum, siginfo_t *info, void *context);
参数说明:
signum:信号的编号。
info:记录信号发送进程信息的结构体。
context:可以赋给指向 ucontext_t 类型的一个对象的指针,以引用在传递信号
时被中断的接收进程或线程的上下文。
示例代码
#include<stdio.h>
#include<signal.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/time.h>void my_func(int sig)
{
printf("ctrl+c被按下了\n");
}
int main()
{
struct sigaction act;
//act存放回调函数
act.sa_handler = my_func;
//act添加阻塞集 act.sa_mask
//清空阻塞集
//sigemptyset(&act.sa_mask);
//将所有信号添加到阻塞集中
sigfillset(&act.sa_mask);
//SA_RESETHAND:信号处理之后重新设置为默认的处理方式
act.sa_flags = 0; //默认方式
//act.sa_flags |= SA_RESETHAND;sigaction(SIGINT, &act, NULL);
while (1);
return 0;
}