Linux 线程信号

2018-07-18  本文已影响0人  0X7C00

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);
}

上一篇下一篇

猜你喜欢

热点阅读