关于sigwait

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

  因而网上找了一些资料,总的理解以下所示:
sigwait(&set, signo)监听信号集set中所包含的信号,并将其存在signo中。注意:sigwait函数所监听的信号在以前必须被阻塞。
sigwait函数将阻塞调用他的线程,直到收到它所监听的信号发生了,而后sigwait将其从未决队列中取出(由于被阻塞了,因此确定是未决了),可是有一点须要注意的是:它从未决队列取出以后,并不影响那个被取出的信号原来被阻塞的状态。它所作的工做只有两个:第一,监听被阻塞的信号;第二,若是所监听的信号产生了,则将其从未决队列中移出来(这里实时信号和非实时信号又有区别,体如今取出的顺序等,具体本身取网上查,这里再也不详述)。在一些帖子中看到:sigwait取出未决信号以后,并将其原来的阻塞状态转为非阻塞状态,这是严重错误的,sigwait并不改变信号的阻塞与非阻塞状态,它只作上面的两个工做。(以上描述有错的话,欢迎指正)
因而就写了一个简单的程序测试了一下:
#include<stdio.h>
#include<pthread.h>
#include<signal.h>

static void sig_alrm(int signo);
static void sig_init(int signo);
int
main()
{
    sigset_t set;
    int sig;
    sigemptyset(&set);
    sigaddset(&set, SIGALRM);
    pthread_sigmask(SIG_SETMASK, &set, NULL);//阻塞SIGALRM信号
    
    signal(SIGALRM, sig_alrm);
    signal(SIGINT, sig_init);
    sigwait(&set, &sig);//sigwait只是从未决队列中删除该信号,并不改变信号掩码。也就是,当sigwait函数返回,它监听的信号依旧被阻塞。
    switch(sig){
    case 14:
        printf("sigwait, receive signal SIGALRM\n");
        /*do the job when catch the sigwait*/
        break;
    default: 
        break;
    }
    sigdelset(&set, SIGALRM);
    pthread_sigmask(SIG_SETMASK, &set, NULL);

    for(;;)
    {}
    return 0;
}

static void
sig_alrm(int signo)
{
    printf("after sigwait, catch SIGALRM\n");
    fflush(stdout);
    return ;
}

static void
sig_init(int signo)
{
    printf("catch SIGINT\n");
    return ;
}

  

在程序中:    
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
相关文章
相关标签/搜索