zoukankan      html  css  js  c++  java
  • 关于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
    ");
            /*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
    ");
        fflush(stdout);
        return ;
    }
    
    static void
    sig_init(int signo)
    {
        printf("catch SIGINT
    ");
        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
  • 相关阅读:
    网络安全分析
    java实现 洛谷 P1464 Function
    java实现 洛谷 P1464 Function
    java实现 洛谷 P1014 Cantor表
    java实现 洛谷 P1014 Cantor表
    java实现 洛谷 P1014 Cantor表
    java实现 洛谷 P1014 Cantor表
    java实现 洛谷 P1014 Cantor表
    java实现 洛谷 P1540 机器
    java实现 洛谷 P1540 机器
  • 原文地址:https://www.cnblogs.com/c-slmax/p/5842293.html
Copyright © 2011-2022 走看看