[学习笔记]信号的高级用法

↘锁芯ラ 提交于 2020-03-15 13:45:33

sigaction函数注册信号处理函数

sigaction函数

q  包含头文件<signal.h>

q  功能:sigaction函数用于改变进程接收到特定信号后的行为。

q  原型:

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

q  参数

q  该函数的第一个参数为信号的值,可以为除SIGKILL及SIGSTOP外的任何一 个特定有效的信号(为这两个信号定义自己的处理函数,将导致信号安装错误)

q  第二个参数是指向结构sigaction的一个实例的指针,在结构 sigaction的实例中,指定了对特定信号的处理,可以为空,进程会以缺省方式对信号处理

q  第三个参数oldact指向的对象用来保存原来对相应信号的处理,可指定oldact为NULL。

q  返回值:函数成功返回0,失败返回-1

    signal(num., handle)

 

sigaction结构体

q  第二个参数最为重要,其中包含了对指定信号的处理、信号所传递的信息、信号处理函数执行过程中应屏蔽掉哪些函数等等

struct sigaction {

         void (*sa_handler)(int);   //信号处理程序 不接受额外数据

         void (*sa_sigaction)(int, siginfo_t *, void *); //信号处理程序 能接受额外数据,和sigqueue配合使用

         sigset_t sa_mask; //

         int sa_flags; //影响信号的行为 SA_SIGINFO表示能接受数据

         void (*sa_restorer)(void); //废弃

};

注意1:回调函数句柄sa_handler、sa_sigaction只能任选其一。

注意2:The sigaction structure is defined as something like 思考如何测试?

会查找、会用man手册,是通往高手的必经之路。

       The siginfo_t parameter to sa_sigaction is a struct with the following elements

 

              siginfo_t {

                  int      si_signo;  /* Signal number */

                  int      si_errno;  /* An errno value */

                  int      si_code;   /* Signal code */

                  pid_t    si_pid;    /* Sending process ID */

                  uid_t    si_uid;    /* Real user ID of sending process */

                  int      si_status; /* Exit value or signal */

                  clock_t  si_utime;  /* User time consumed */

                  clock_t  si_stime;  /* System time consumed */

                  sigval_t si_value;  /* Signal value */

                  int      si_int;    /* POSIX.1b signal */

                  void *   si_ptr;    /* POSIX.1b signal */

                  void *   si_addr;   /* Memory location which caused fault */

                  int      si_band;   /* Band event */

                  int      si_fd;     /* File descriptor */

              }

实验1sigaction的函数注册信号,基本用法

void handler(int sig)

{

         printf("recv a sig=%d\n", sig);

}

 

__sighandler_t my_signal(int sig, __sighandler_t handler)

{

         struct sigaction act;

         struct sigaction oldact;

         act.sa_handler = handler;

         sigemptyset(&act.sa_mask);

         act.sa_flags = 0;

 

         if (sigaction(sig, &act, &oldact) < 0)

                   return SIG_ERR;

 

         return oldact.sa_handler;

}

 

int main(int argc, char *argv[])

{        

         struct sigaction act;

         sigset_t sa_mask;

        

         act.sa_handler = handler;

         act.sa_flags = 0;

         sigemptyset(&act.sa_mask);

 

         //测试信号安装函数

         //sigaction(SIGINT, &act, NULL);

        

         //模拟signal函数

         my_signal(SIGINT, handler);

 

         for (;;)

         {

                   pause();

         }

         return 0;

}

 

实验2:测试sigaction结构体第三个参数sigset_t sa_mask的作用

/*

struct sigaction {

      void (*sa_handler)(int);

      void (*sa_sigaction)(int, siginfo_t *, void *);

      sigset_t sa_mask; 

      int sa_flags; 

      void (*sa_restorer)(void);

  } */

//测试sigaction结构体第三个参数sigset_t sa_mask的作用

//作用 sigaddset(&act.sa_mask, SIGQUIT); 加入到sa_mask中的信号,被阻塞(信号处理函数执行的过程中被阻塞)。

//注意:SIGQUIT信号最终还会抵达

 

int main(int argc, char *argv[])

{

         struct sigaction act;

         act.sa_handler = handler;

        

         sigemptyset(&act.sa_mask);

         sigaddset(&act.sa_mask, SIGQUIT);

         act.sa_flags = 0;

 

         if (sigaction(SIGINT, &act, NULL) < 0)

                   ERR_EXIT("sigaction error");

 

         for (;;)

                   pause();

         return 0;

}

 

void handler(int sig)

{

         printf("recv a sig=%d 信号处理函数执行的时候,阻塞sa_mask中的信号\n", sig);

         sleep(5);

}

 

sigqueue新的信号发送函数

sigqueue函数

q  功能:新的发送信号系统调用,主要是针对实时信号提出的支持信号带有参数,与函数sigaction()配合使用。

q  注意:和kill函数相比Int kill(pid_t pid, int siq)多了参数

q  原型:

         int sigqueue(pid_t pid, int sig, const union sigval value);

q  参数

q   sigqueue的第1个参数是指定接收信号的进程id,第2个参数确定即将发送的信号,第3个参数是一个联合数据结构union sigval,指定了信号传递的参数,即通常所说的4字节值。

q  返回值成功返回0,失败返回-1

q  sigqueue()比kill()传递了更多的附加信息,但sigqueue()只能向一个进程发送信号,而不能发送信号给一个进程组。

q  sigval联合体

typedef union sigval

           {

                     int sival_int;

                      void *sival_ptr;

}sigval_t;

 

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!