sigaction函数的使用

sigaction函数的使用sigaction函数的功能是检查或修改与指定信号相关联的处理动作(可同时两种操作)。他是POSIX的信号接口,而signal()是标准C的信号接口(如果程序必须在非POSIX系统上运行,那么就应该使用这个接口)给信号signum设置新的信号处理函数act,同时保留该信号原有的信号处理函数oldactintsigaction(intsigno,conststruct

大家好,又见面了,我是你们的朋友全栈君。

sigaction函数的功能是检查或修改与指定信号相关联的处理动作(可同时两种操作)。
他是POSIX的信号接口,而signal()是标准C的信号接口(如果程序必须在非POSIX系统上运行,那么就应该使用这个接口)
给信号signum设置新的信号处理函数act, 同时保留该信号原有的信号处理函数oldact
int sigaction(int signo,const struct sigaction *restrict act,
              struct sigaction *restrict oact);
结构sigaction定义如下:
struct sigaction{
  void (*sa_handler)(int);
   sigset_t sa_mask;
  int sa_flag;
  void (*sa_sigaction)(int,siginfo_t *,void *);
};
sa_handler字段包含一个信号捕捉函数的地址
sa_mask字段说明了一个信号集,在调用该信号捕捉函数之前,这一信号集要加进进程的信号屏蔽字中。仅当从信号捕捉函数返回时再将进程的信号屏蔽字复位为原先值。
sa_flag是一个选项,主要理解两个
SA_INTERRUPT 由此信号中断的系统调用不会自动重启
SA_RESTART 由此信号中断的系统调用会自动重启
SA_SIGINFO 提供附加信息,一个指向siginfo结构的指针以及一个指向进程上下文标识符的指针
最后一个参数是一个替代的信号处理程序,当设置SA_SIGINFO时才会用他。
例子:
#include
#include
#include
void show_handler(int sig)
{
    printf(“I got signal %d\n”, sig);
    int i;
    for(i = 0; i < 5; i++) {
        printf(“i = %d\n”, i);
        sleep(1);
    }
}
 
int main(void)
{
    int i = 0;
    struct sigaction act, oldact;
    act.sa_handler = show_handler;
    sigaddset(&act.sa_mask, SIGQUIT); //见注(1)
    act.sa_flags = SA_RESETHAND | SA_NODEFER; //见注(2)
    //act.sa_flags = 0; //见注(3)
 
    sigaction(SIGINT, &act, &oldact);
    while(1) {
        sleep(1);
        printf(“sleeping %d\n”, i);
        i++;
    }
}
 
注:
(1)    如果在信号SIGINT(Ctrl + c)的信号处理函数show_handler执行过程中,本进程收到信号SIGQUIT(Crt+\),将阻塞该信号,直到show_handler执行结束才会处理信号SIGQUIT。
 
(2)    SA_NODEFER       一般情况下, 当信号处理函数运行时,内核将阻塞<该给定信号 — SIGINT>。但是如果设置了SA_NODEFER标记, 那么在该信号处理函数运行时,内核将不会阻塞该信号。 SA_NODEFER是这个标记的正式的POSIX名字(还有一个名字SA_NOMASK,为了软件的可移植性,一般不用这个名字)    
       SA_RESETHAND    当调用信号处理函数时,将信号的处理函数重置为缺省值。 SA_RESETHAND是这个标记的正式的POSIX名字(还有一个名字SA_ONESHOT,为了软件的可移植性,一般不用这个名字)   
 
(3)    如果不需要重置该给定信号的处理函数为缺省值;并且不需要阻塞该给定信号(无须设置sa_flags标志),那么必须将sa_flags清零,否则运行将会产生段错误。但是sa_flags清零后可能会造成信号丢失!
 

sigaction

 
  sigaction(查询或设置信号处理方式)

 

  相关函数 signal,sigprocmask,sigpending,
sigsuspend, sigemptyset

 

  表头文件 #include

 

  定义函数 int sigaction(int signum,const struct sigaction *act ,struct sigaction *oldact);

 

  函数说明 sigaction()会依参数signum指定的信号编号来设置该信号的处理函数。参数signum可以指定SIGKILL和SIGSTOP以外的所有信号。

 

  如参数结构sigaction定义如下

 

  struct sigaction {

 

  union{

 

  __sighandler_t sa_handler;

 

  void (*_sa_sigaction)(int,struct siginfo *, void *);

 

  }_u

 

  sigset_t sa_mask;

 

  unsigned long sa_flags;

 

  void (*sa_restorer)(void);

 

  }

 

  信号处理函数可以采用void (*sa_handler)(int)或void (*sa_sigaction)(int, siginfo_t *, void *)。到底采用哪个要看sa_flags中是否设置了SA_SIGINFO位,如果设置了就采用void (*sa_sigaction)(int, siginfo_t *, void *),此时可以向处理函数发送附加信息;默认情况下采用void (*sa_handler)(int),此时只能向处理函数发送信号的数值。

 

  sa_handler此参数和signal()的参数handler相同,代表新的信号处理函数,其他意义请参考signal()。

 

  sa_mask 用来设置在处理该信号时暂时将sa_mask 指定的信号搁置。

 

  sa_restorer 此参数没有使用。

 

  sa_flags 用来设置信号处理的其他相关操作,下列的数值可用。 

 

  sa_falgs还可以设置其他标志:

 

  SA_RESETHAND:当调用信号处理函数时,将信号的处理函数重置为缺省值SIG_DFL

 

  ··SA_RESTART:如果信号中断了进程的某个系统调用,则系统自动启动该系统调用

 

  SA_NODEFER :一般情况下, 当信号处理函数运行时,内核将阻塞该给定信号。但是如果设置了 SA_NODEFER标记, 那么在该信号处理函数运行时,内核将不会阻塞该信号

 

  sigaction

sigaction()

示例

  #include

 

  static void int_hander(int s)

  {

  printf(“Catch a signal sigint\n”);

  }

 

  int

 

  main(void)

 

  {

 

  int i;

 

  struct sigaction act, oact;

 

  act. sa_handler = int_hander;

 

  sigemptyset(&act. sa_mask); //清空此信号集

 

  act. sa_flags = 0;

 

  sigaction(SIGINT, &act, &oact);

 

  //signal(SIGINT, SIG_IGN);

 

  while(1){

 

  for(i=0; i<5; i++){

 

  write(1, “.”, 1);

 

  sleep(1);

 

  }

 

  write(1, “\n”, 1);

 

  }

 

  sigaction(SIGINT, &oact, NULL); //恢复成原始状态

 

  return 0;

 

  }

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/141428.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • win10更改计算机时间格式,Win10电脑锁屏的时钟格式怎么更改[通俗易懂]

    win10更改计算机时间格式,Win10电脑锁屏的时钟格式怎么更改[通俗易懂]Win10电脑锁屏的时钟格式怎么更改腾讯视频/爱奇艺/优酷/外卖充值4折起我们在使用Win10电脑的锁定屏幕后,默认显示的是12小时的时钟格式。想要将12小时格式设置为24小时格式,该怎么操作呢?今天就跟大家介绍一下Win10电脑锁屏的时钟格式怎么更改的具体操作步骤。1.首先打开电脑上的“控制面板”,进入页面后,找到“更改日期,时间或数字格式”选项。2.如图,在弹出的窗口中,如果想要设置为2…

  • 振荡周期、时钟周期、机器周期、指令周期的区别与联系[通俗易懂]

    振荡周期、时钟周期、机器周期、指令周期的区别与联系[通俗易懂]以下内容均来自网上查找,并根据个人理解进行整理,刚开始学习单片机,如有不对的地方敬请指正。先给出结论:一个振荡周期=一个时钟周期;一个时钟周期=一个机器周期;一个机器周期=六个状态周期;一个状态周期=两个节拍;一个节拍=一个时钟周期;一个指令周期=N个机器周期;综上:1个指令周期=N个机器周期=6N个状态周期=12N个节拍=12N个时钟周期=12N个振荡周期时钟周期:一个脉冲所需…

    2022年10月13日
  • 端口号大全「建议收藏」

    端口号大全「建议收藏」计算机系统“端口”是英文port的义译,可以认为是计算机与外界通讯交流的出口。其中硬件领域的端口又称接口,如:USB端口、串行端口、打印机端口等。软件领域的端口一般指网络中面向连接服务和无连接服务的通信协议端口,是一种抽象的软件结构,包括一些数据结构和I/O(基本输入输出)缓冲区。说白了,我们在计算机的任何操作都在有意无意的使用着计算机的各个端口,下面列出了计算机的常用端口:0…

  • c#语言简介_简单介绍自己

    c#语言简介_简单介绍自己taskScheduler根据定义ThetaskSchedulerbythedefinitionblurb.“Istheclasswheretheusagecontextiswithinthetasklibraries.“它的作用像是WPF/Winform时代的SynchronizationContext.ItisliketheSync…

  • 小明の魔法计划——最长上升子序列[通俗易懂]

    小明の魔法计划——最长上升子序列[通俗易懂]Think:1知识点:最长上升子序列2反思:知识体系需要加深拓展SDUT题目链接小明の魔法计划TimeLimit:1000MSMemoryLimit:65536KBProblemDescription在一个遥远的数学魔法国度,小明在学习一个魔法,这个魔法需要一些施法材料,所幸的是施法材料已经准备好了,下一步就是建立魔法阵了,每一个施法材料都有一个特性值,表示为一个大于1小

  • goland激活码20213月最新在线激活

    goland激活码20213月最新在线激活,https://javaforall.cn/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号