Chinaunix首页 | 论坛 | 博客
  • 博客访问: 202529
  • 博文数量: 81
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 540
  • 用 户 组: 普通用户
  • 注册时间: 2014-05-04 12:48
文章分类
文章存档

2015年(23)

2014年(58)

我的朋友

分类: C/C++

2015-02-04 12:03:02

刚开始看sigwait函数,只是知道它是用来解除阻塞的信号,可是使我疑惑的是那么解除了以后为什么线程收到终止信号SIGINT的时候还是没能终止呢?
  于是网上找了一些资料,总的理解如下所示:
sigwait(&set, signo)监听信号集set中所包含的信号,并将其存在signo中。注意:sigwait函数所监听的信号在之前必须被阻塞。
sigwait函数将阻塞调用他的线程,直到收到它所监听的信号发生了,然后sigwait将其从未决队列中取出(因为被阻塞了,所以肯定是未决了),但是有一点需要注意的是:它从未决队列取出之后,并不影响那个被取出的信号原来被阻塞的状态。它所做的工作只有两个:第一,监听被阻塞的信号;第二,如果所监听的信号产生了,则将其从未决队列中移出来(这里实时信号和非实时信号又有区别,体现在取出的顺序等,具体自己取网上查,这里不再详述)。在一些帖子中看到:sigwait取出未决信号之后,并将其原来的阻塞状态转为非阻塞状态,这是严重错误的,sigwait并不改变信号的阻塞与非阻塞状态,它只做上面的两个工作。(以上描述有错的话,欢迎指正)
于是就写了一个简单的程序测试了一下:

点击(此处)折叠或打开

  1. #include
  2. #include
  3. #include

  4. static void sig_alrm(int signo);
  5. static void sig_init(int signo);
  6. int
  7. main()
  8. {
  9.     sigset_t set;
  10.     int sig;
  11.     sigemptyset(&set);
  12.     sigaddset(&set, SIGALRM);
  13.     pthread_sigmask(SIG_SETMASK, &set, NULL);//阻塞SIGALRM信号
  14.     
  15.     signal(SIGALRM, sig_alrm);
  16.     signal(SIGINT, sig_init);
  17.     sigwait(&set, &sig);//sigwait只是从未决队列中删除该信号,并不改变信号掩码。也就是,当sigwait函数返回,它监听的信号依旧被阻塞。
  18.     switch(sig){
  19.     case 14:
  20.         printf("sigwait, receive signal SIGALRM\n");
  21.         /*do the job when catch the sigwait*/
  22.         break;
  23.     default: 
  24.         break;
  25.     }
  26.     sigdelset(&set, SIGALRM);
  27.     pthread_sigmask(SIG_SETMASK, &set, NULL);

  28.     for(;;)
  29.     {}
  30.     return 0;
  31. }

  32. static void
  33. sig_alrm(int signo)
  34. {
  35.     printf("after sigwait, catch SIGALRM\n");
  36.     fflush(stdout);
  37.     return ;
  38. }

  39. static void
  40. sig_init(int signo)
  41. {
  42.     printf("catch SIGINT\n");
  43.     return ;
  44. }
在程序中:    
sigdelset(&set, SIGALRM);
pthread_sigmask(SIG_SETMASK, &set, NULL);
上面两句如果不加的话,那么SIGALRM将一直被阻塞,我连续发送了4次KILL -14  17223(进程号)给测试进程,只有第一次会打印switch里面的语句:
sigwait, receive signal SIGALRM 。
后面发送的信号将被阻塞。由于被阻塞,所以信号处理程序无法捕捉信号,故之后发送信号不会有任何输出。
当加了上面两句话以后,第一次发送kill -14 17457(进程号)时打印switch里面的语句:
sigwait, receive signal SIGALRM
之后发送SIGALRM信号的话将被信号处理程序捕捉。
after sigwait, catch SIGALRM
after sigwait, catch SIGALRM
after sigwait, catch SIGALRM
阅读(977) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~