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