美文网首页
Linux 线程信号

Linux 线程信号

作者: 0X7C00 | 来源:发表于2018-07-18 22:10 被阅读0次

    1. 线程信号原理

    在多线程环境下,信号处理函数是多个线程共有的,若一个线程修改信号处理函数,则所有的线程均会受到影响。

    每个线程可以有不同的信号屏蔽字,当一个信号发生时,若该信号与定时器相关,则信号送给该信号产生的线程中,对于其他信号Linux内核随机选择一个线程进行递送。

    当多个线程在调用sigwait等待同一信号时,只有一个线程会返回,其他线程继续阻塞。
    当进程有针对某一信号有sigaction函数,同时线程也在等待该信号的发生,则该信号递送给线程还是进程是不确定的。

    2. sigwait的好处

    sigwait的功能是等待一些信号的发生,在调用该函数钱需要阻塞sigwait等待的信号,
    在调用sigwait的过程中,线程会解除对写着信号的阻塞,即信号可以被递送到线程中。

    在进程中信号的处理是异步的,信号随时有可能发生,会中断正在执行的指令,需要保证信号处理函数是可重入的(即信号处理函数是异步信号安全函数)。

    sigwait的好处在于该函数可以把异步变成同步,主动等待信号的发生,同步的好处在于不必要求信号处理函数是异步信号安全的。

    3. pthread_sigmask与sigprocmask如何选择

    在多线程环境中sigprocmask函数的行为是未定义的,在多线程的环境中信号屏蔽需要使用pthread_sigmask函数

    4. 检查一个线程是否存在

    检查一个线程是否存在可以调用pthread_kill函数,该函数的声明为:

    int pthread_kill(pthread_t thread, int sig);
    

    当发送的信号为0时可以用来检查线程是否存在。
    下面的代码用来演示pthread_kill检查线程是否存在

    #include <stdio.h>
    #include <stdlib.h>
    #include <pthread.h>
    #include <errno.h>
    #include <unistd.h>
    #include <signal.h>
    void *func1()/*1秒钟之后退出*/
    {
        sleep(1);
        printf("线程1(ID:0x%x)退出。/n",(unsigned int)pthread_self());
        pthread_exit((void *)0);
    }
    void *func2()/*5秒钟之后退出*/
    {
        sleep(5);
        printf("线程2(ID:0x%x)退出。/n",(unsigned int)pthread_self());
        pthread_exit((void *)0);
    }
    void test_pthread(pthread_t tid) /*pthread_kill的返回值:成功(0) 线程不存在(ESRCH) 信号不合法(EINVAL)*/
    {
        int pthread_kill_err;
        pthread_kill_err = pthread_kill(tid,0);
        if(pthread_kill_err == ESRCH)
            printf("ID为0x%x的线程不存在或者已经退出。/n",(unsigned int)tid);
        else if(pthread_kill_err == EINVAL)
            printf("发送信号非法。/n");
        else
            printf("ID为0x%x的线程目前仍然存活。/n",(unsigned int)tid);
    }
    int main()
    {
        int ret;
        pthread_t tid1,tid2;
        pthread_create(&tid1,NULL,func1,NULL);
        pthread_create(&tid2,NULL,func2,NULL);
        sleep(3);/*创建两个进程3秒钟之后,分别测试一下它们是否还活着*/
        test_pthread(tid1);/*测试ID为tid1的线程是否存在*/
        test_pthread(tid2);/*测试ID为tid2的线程是否存在*/
        exit(0);
    }
    
    

    相关文章

      网友评论

          本文标题:Linux 线程信号

          本文链接:https://www.haomeiwen.com/subject/egfmpftx.html