sigaction函数

作者: 奥斯特洛司机 | 来源:发表于2019-03-28 10:50 被阅读0次

    该函数补充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(;;);
    }

    相关文章

      网友评论

        本文标题:sigaction函数

        本文链接:https://www.haomeiwen.com/subject/cyvfbqtx.html