1. 程式人生 > >Linux多執行緒程式設計時如何檢視一個程序中的某個執行緒是否存活

Linux多執行緒程式設計時如何檢視一個程序中的某個執行緒是否存活

pthread_kill:

別被名字嚇到,pthread_kill可不是kill,而是向執行緒傳送signal。還記得signal嗎,大部分signal的預設動作是終止程序的執行,所以,我們才要用signal()去抓訊號並加上處理函式。

int pthread_kill(pthread_t thread, int sig);

向指定ID的執行緒傳送sig訊號,如果執行緒程式碼內不做處理,則按照訊號預設的行為影響整個程序,也就是說,如果你給一個執行緒傳送了SIGQUIT,但執行緒卻沒有實現signal處理函式,則整個程序退出。

pthread_kill(threadid, SIGKILL)也一樣,殺死整個程序。
如果要獲得正確的行為,就需要線上程內實現signal(SIGKILL,sig_handler)了。

所以,如果int sig的引數不是0,那一定要清楚到底要幹什麼,而且一定要實現執行緒的訊號處理函式,否則,就會影響整個程序。

OK,如果int sig是0呢,這是一個保留訊號,一個作用是用來判斷執行緒是不是還活著。

我們來看一下pthread_kill的返回值:
成功:0
執行緒不存在:ESRCH
訊號不合法:EINVAL

所以,pthread_kill(threadid,0)就很有用啦。

int kill_rc = pthread_kill(thread_id,0);

if(kill_rc == ESRCH)
printf("the specified thread did not exists or already quit/n");
else if(kill_rc == EINVAL)
printf("signal is invalid/n");
else
printf("the specified thread is alive/n");

上述的程式碼就可以判斷執行緒是不是還活著了。 

使用pthread_kill函式檢測一個執行緒是否還活著的程式,在linux環境下gcc編譯通過,現將程式碼貼在下面:

/******************************* pthread_kill.c *******************************/
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <errno.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);
}


編譯:gcc -o pthread_kill -lpthread pthread_kill.c
執行:./pthread_kill

///////////////////////// 執行結果 /////////////////////////////
執行緒1(ID:0xb7e95

b90)退出。
ID為0xb7e95b90的執行緒不存在或者已經退出。
ID為0xb7694b90的執行緒目前仍然存活。