Linux系统编程:signal、sigaction、sigpending、sigprocmask函数


一、signal

作用:设置信号处理方式,当进程接收到信号时,指定特定的handler方法来处理信号(只起一次作用)。

头文件     
    #include<signal.h>
定义函数
    void (*signal(int signum,void(* handler)(int)))(int);
函数说明
    signal()会依参数signum 指定的信号编号来设置该信号的处理函数。当指定的信号到达时就会跳转到参数handler指定的函数执行。如果参数handler不是函数指针,则必须是下列两个常数之一:
    SIG_IGN 忽略参数signum指定的信号。
    SIG_DFL 将参数signum 指定的信号重设为核心预设的信号处理方式。
返回值
    返回先前的信号处理函数指针,如果有错误则返回SIG_ERR(-1)。
附加说明
    在信号发生跳转到自定的handler处理函数执行后,系统会自动将此处理函数换回原来系统预设的处理方式,如果要改变此操作请改用sigaction()。

示例代码:

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <string.h>
#include <signal.h>

void my_func(int sign_no)

{

    if (sign_no == SIGINT)

        printf("I have get SIGINT\n");

    else if (sign_no == SIGQUIT)

        printf("I have get SIGQUIT\n");
}

int main()

{

    printf("Waiting for signal SIGINT or SIGQUIT \n ");

    /*发出相应的信号,并跳转到信号处理函数处*/

    signal(SIGINT, my_func);

    signal(SIGQUIT, my_func);

    pause();

    pause();

    exit(0);
}

二、 sigaction

相关函数

signal,sigprocmask,sigpending,sigsuspend

表头文件

#include<signal.h>

定义函数

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

函数说明

sigaction()会依参数signum指定的信号编号来设置该信号的处理函数。参数signum可以指定SIGKILL和SIGSTOP以外的所有信号。
如参数结构sigaction定义如下
struct sigaction
{
void (*sa_handler) (int);
sigset_t sa_mask;
int sa_flags;
void (*sa_restorer) (void);
}
sa_handler此参数和signal()的参数handler相同,代表新的信号处理函数,其他意义请参考signal()。
sa_mask 用来设置在处理该信号时暂时将sa_mask 指定的信号搁置。
sa_restorer 此参数没有使用。
sa_flags 用来设置信号处理的其他相关操作,下列的数值可用。
OR 运算(|)组合
A_NOCLDSTOP : 如果参数signum为SIGCHLD,则当子进程暂停时并不会通知父进程
SA_ONESHOT/SA_RESETHAND:当调用新的信号处理函数前,将此信号处理方式改为系统预设的方式。
SA_RESTART:被信号中断的系统调用会自行重启
SA_NOMASK/SA_NODEFER:在处理此信号未结束前不理会此信号的再次到来。
如果参数oldact不是NULL指针,则原来的信号处理方式会由此结构sigaction 返回。

返回值

执行成功则返回0,如果有错误则返回-1。

错误代码

EINVAL 参数signum 不合法, 或是企图拦截SIGKILL/SIGSTOPSIGKILL信号
EFAULT 参数act,oldact指针地址无法存取。
EINTR 此调用被中断

范例

#include<unistd.h>
#include<signal.h>
void show_handler(struct sigaction * act)
{
switch (act->sa_flags)
{
case SIG_DFL:printf(“Default action\n”);break;
case SIG_IGN:printf(“Ignore the signal\n”);break;
default: printf(“0x%x\n”,act->sa_handler);
}
}
main()
{
int i;
struct sigaction act,oldact;
act.sa_handler = show_handler;
act.sa_flags = SA_ONESHOT|SA_NOMASK;
sigaction(SIGUSR1,&act,&oldact);
for(i=5;i<15;i++)
{
printf(“sa_handler of signal %2d =”.i);
sigaction(i,NULL,&oldact);
}
}

执行

sa_handler of signal 5 = Default action
sa_handler of signal 6= Default action
sa_handler of signal 7 = Default action
sa_handler of signal 8 = Default action
sa_handler of signal 9 = Default action
sa_handler of signal 10 = 0x8048400
sa_handler of signal 11 = Default action
sa_handler of signal 12 = Default action
sa_handler of signal 13 = Default action
sa_handler of signal 14 = Default action

三、 sigpending

函数作用查询被搁置的信号

相关函数

signal,sigaction,sigprocmask,sigsuspend

表头文件

#include<signal.h>

定义函数

int sigpending(sigset_t *set);

函数说明

sigpending()会将被搁置的信号集合由参数set指针返回。

返回值执

行成功则返回0,如果有错误则返回-1。

错误代码

EFAULT 参数set指针地址无法存取
EINTR 此调用被中断。

示例代码:

#include <stdio.h>
#include <signal.h>
#include <stdlib.h>

int main(int argc, const char *argv[])
{
    sigset_t my_set;
    sigpending(&my_set);
    // 每个一秒读取一次当前进程的未决信号集
    while (1)
    {
        int i;
        for (i = 1; i < 32; ++i)
        {   //对每一个信号进行一次判断
            if (sigismember(&my_set, i))
            {
                printf("1");
            }
            else
            {
                printf("0");
            }
        }
        printf("\n");
        sleep(1);
    }
}

四、 sigprocmask

函数作用查询或设置信号遮罩

相关函数

signal,sigaction,sigpending,sigsuspend

表头文件

#include<signal.h>

定义函数

int sigprocmask(int how,const sigset_t *set,sigset_t * oldset);

函数说明

sigprocmask()可以用来改变目前的信号遮罩,其操作依参数how来决定
SIG_BLOCK 新的信号遮罩由目前的信号遮罩和参数set 指定的信号遮罩作联集
SIG_UNBLOCK 将目前的信号遮罩删除掉参数set指定的信号遮罩
SIG_SETMASK 将目前的信号遮罩设成参数set指定的信号遮罩。
如果参数oldset不是NULL指针,那么目前的信号遮罩会由此指针返回。

返回值

执行成功则返回0,如果有错误则返回-1。

错误代码

EFAULT 参数set,oldset指针地址无法存取。
EINTR 此调用被中断

 

示例代码:

#include <stdio.h>
#include <signal.h>
#include <stdlib.h>

int main(int argc, const char *argv[])
{
    //自定义信号集
    sigset_t my_set;
    //手动清空信号集
    sigemptyset(&my_set);
    // 往信号集中添加要阻塞的信号
    sigaddset(&my_set, SIGINT);
    sigaddset(&my_set, SIGQUIT);
    sigaddset(&my_set, SIGKILL);
    // 将自定义的集合数据设置给内核的阻塞信号集 
    sigprocmask(SIG_BLOCK, &my_set, NULL);
    // 每隔一秒读取一次内存中的未决信号集
    while (1)
    {
        int i;
        for (i = 1; i < 32; ++i)
        { //对每一个信号进行一次判断
            if (sigismember(&my_set, i))
            {
                printf("1");
            }
            else
            {
                printf("0");
            }
        }
        printf("\n");
        sleep(1);
    }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值