linux中使用信号--sigwait()和pthread_sigmask()

1. sigwait函数:
sigwait等一个或者多个指定信号发生。
    它所做的工作只有两个:第一,监听被阻塞的信号;第二,如果所监听的信号产生了,则将其从未决队列中移出来(这里实时信号和非实时信号又有区别,体现在取出的顺序 等,具体自己取网上查,这里不再详述)。 sigwait并不改变信号掩码的阻塞与非阻塞状态。
     在POSIX标准中,当进程收到信号时,如果是多线程的情况,我们是无法确定是哪一个线程处理这个信号。而sigwait是从进程中pending的信号中,取走指定的信号。这样的 话,如果要确保sigwait这个线程收到该信号,那么所有线程含主线程以及这个sigwait线程则必须block住这个信号,因为如果自己不阻塞就没有未决状态(阻塞状态)信号,别的所有线程不阻塞就有可能当信号过来时,被其他的线程处理掉。
记住:
     在多线程代码中,总是使用sigwait或者sigwaitinfo或者sigtimedwait等函数来处理信号。
     而不是signal或者sigaction等函数。因为在一个线程中调用signal或者sigaction等函数会改变所以线程中的信号处理函数,而不是仅仅改变调用signal/sigaction的那个线程的信号处理函数。

2. pthread_sigmask函数:
   每个线程均有自己的信号屏蔽集(信号掩码),可以使用pthread_sigmask函数来屏蔽某个线程对某些信号的
   响应处理,仅留下需要处理该信号的线程来处理指定的信号。实现方式是:利用线程信号屏蔽集的继承关系
  (在主进程中对sigmask进行设置后,主进程创建出来的线程将继承主进程的掩码)

3. pthread_kill函数:
   在多线程程序中,一个线程可以使用pthread_kill对同一个进程中指定的线程(包括自己)发送信号。注意在多线程中  
  一般不使用kill函数发送信号,因为kill是对进程发送信号,结果是:正在运行的线程会处理该信号,如果该线程没有
 注册信号处理函数,那么会导致整个进程退出。

记住:调用sigwait同步等待的信号必须在调用线程中被屏蔽,并且通常应该在所有的线程中被屏蔽(这样可以保证信号绝不会被送到除了调用sigwait的任何其它线程),这是通过利用信号掩码的继承关系来达到的。


刚开始看sigwait函数,只是知道它是用来解除阻塞的信号,可是使我疑惑的是那么解除了以后为什么线程收到终止信号SIGINT的时候还是没能终止呢?

  于是网上找了一些资料,总的理解如下所示:
sigwait(&set, signo)监听信号集set中所包含的信号,并将其存在signo中。注意:sigwait函数所监听的信号在之前必须被阻塞。
sigwait函数将阻塞调用他的线程,直到收到它所监听的信号发生了,然后sigwait将其从未决队列中取出(因为被阻塞了,所以肯定是未决了),但是有一点需要注意的是:它从未决队列取出之后,并不影响那个被取出的信号原来被阻塞的状态。它所做的工作只有两个:第一,监听被阻塞的信号;第二,如果所监听的信号产生了,则将其从未决队列中移出来(这里实时信号和非实时信号又有区别,体现在取出的顺序等,具体自己取网上查,这里不再详述)。在一些帖子中看到:sigwait取出未决信号之后,并将其原来的阻塞状态转为非阻塞状态,这是严重错误的,sigwait并不改变信号的阻塞与非阻塞状态,它只做上面的两个工作。(以上描述有错的话,欢迎指正)
[cpp]  view plain  copy
  1. #include<stdio.h>  
  2. #include<pthread.h>  
  3. #include<signal.h>  
  4.   
  5. static void sig_alrm(int signo);  
  6. static void sig_init(int signo);  
  7. int  
  8. main()  
  9. {  
  10.     sigset_t set;  
  11.     int sig;  
  12.     sigemptyset(&set);  
  13.     sigaddset(&set, SIGALRM);  
  14.     pthread_sigmask(SIG_SETMASK, &set, NULL);//阻塞SIGALRM信号  
  15.       
  16.     signal(SIGALRM, sig_alrm);  
  17.     signal(SIGINT, sig_init);  
  18.     sigwait(&set, &sig);//sigwait只是从未决队列中删除该信号,并不改变信号掩码。也就是,当sigwait函数返回,它监听的信号依旧被阻塞。  
  19.     switch(sig){  
  20.     case 14:  
  21.         printf("sigwait, receive signal SIGALRM\n");  
  22.         /*do the job when catch the sigwait*/  
  23.         break;  
  24.     default:  
  25.         break;  
  26.     }  
  27.     sigdelset(&set, SIGALRM);  
  28.     pthread_sigmask(SIG_SETMASK, &set, NULL);  
  29.   
  30.     for(;;)  
  31.     {}  
  32.     return 0;  
  33. }  
  34.   
  35. static void  
  36. sig_alrm(int signo)  
  37. {  
  38.     printf("after sigwait, catch SIGALRM\n");  
  39.     fflush(stdout);  
  40.     return ;  
  41. }  
  42.   
  43. static void  
  44. sig_init(int signo)  
  45. {  
  46.     printf("catch SIGINT\n");  
  47.     return ;  
  48. }  
在程序中:    
sigdelset(&set, SIGALRM);
pthread_sigmask(SIG_SETMASK, &set, NULL);
上面两句如果不加的话,那么SIGALRM将一直被阻塞(注意这里所说的阻塞并不是说接受到SIGALARM信号以后仍然还在等待,真的想这样的话,需要将sigwait这个函数放在循环中,这里的意识是说,当这个信号被处理后,如果没有上面的那两句,发送SIGALARM信号仍然没有回应的),我连续发送了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
原文地址:http://blog.csdn.net/yusiguyuan/article/details/14230719
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值