优选主流主机商
任何主机均需规范使用

Linux下高效管理:详解线程取消与终止的实用技巧

关键:

pthread_cancel函数发送终止信号
pthread_setcancelstate函数设置终止方式
pthread_testcancel函数取消线程(另一功能是:设置取消点)

1 线程取消的定义

一般情况下,线程在其主体函数退出的时候会自动终止,但同时也可以因为接收到另一个线程发来的终止(取消)请求而强制终止。

2 线程取消的语义

线程取消的方法是向目标线程发Cancel信号(pthread_cancel函数发送Cancel信号),但如何处理Cancel信号则由目标线程自己决定,或者忽略、或者立即终止、或者继续运行至Cancelation-point(取消点),由不同的Cancelation状态(pthread_setcancelstate函数设置状态)决定。

线程接收到CANCEL信号的缺省处理(即pthread_create()创建线程的缺省状态)是继续运行至取消点,也就是说设置一个CANCELED状态,线程继续运行,只有运行至Cancelation-point的时候才会退出。

3 取消点

根据POSIX标准,pthread_join()、pthread_testcancel()、pthread_cond_wait()、 pthread_cond_timedwait()、sem_wait()、sigwait()等函数以及read()、write()等会引起阻塞的系统调用都是Cancelation-point,而其他pthread函数都不会引起Cancelation动作。但是pthread_cancel的手册页声称,由于LinuxThread库与C库结合得不好,因而目前C库函数都不是Cancelation-point;但CANCEL信号会使线程从阻塞的系统调用中退出,并置EINTR错误码,因此可以在需要作为Cancelation-point的系统调用前后调用 pthread_testcancel(),从而达到POSIX标准所要求的目标,即如下代码段:

pthread_testcancel();
retcode = read(fd, buffer, length);
pthread_testcancel();

4 程序设计方面的考虑

如果线程处于无限循环中,且循环体内没有执行至取消点的必然路径,则线程无法由外部其他线程的取消请求而终止。因此在这样的循环体的必经路径上应该加入pthread_testcancel()调用。

5 与线程取消相关的pthread函数

int pthread_cancel(pthread_t thread)

发送终止信号给thread线程,如果成功则返回0,否则为非0值。发送成功并不意味着thread会终止。

int pthread_setcancelstate(int state, int *oldstate)

设置本线程对Cancel信号的反应,state有两种值:PTHREAD_CANCEL_ENABLE(缺省)和 PTHREAD_CANCEL_DISABLE,分别表示收到信号后设为CANCLED状态和忽略CANCEL信号继续运行;old_state如果不为 NULL则存入原来的Cancel状态以便恢复。

int pthread_setcanceltype(int type, int *oldtype)

设置本线程取消动作的执行时机,type由两种取值:PTHREAD_CANCEL_DEFFERED和 PTHREAD_CANCEL_ASYCHRONOUS,仅当Cancel状态为Enable时有效,分别表示收到信号后继续运行至下一个取消点再退出和立即执行取消动作(退出);oldtype如果不为NULL则存入运来的取消动作类型值。

void pthread_testcancel(void)

功能一:设置取消点;

功能二:检查本线程是否处于Canceld状态,如果是,则进行取消动作,否则直接返回。

代码:

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 #include < stdio.h > #include < errno.h > #include < unistd.h > #include < stdlib.h > #include < pthread.h >     #define THREAD_MAX 4     pthread_mutex_t mutex; pthread_t thread[THREAD_MAX];     static int tries; static int started;     void print_it(int *arg) { pthread_t tid; tid = pthread_self(); printf("Thread %lx was canceled on its %d try.\n",tid,*arg); }     void *Search_Num(int arg) { pthread_t tid; int num; int k=0,h=0,j; int ntries; tid = pthread_self();   /*while(pthread_mutex_trylock(&mutex) == EBUSY) { printf("**************busy****************\n"); pthread_testcancel(); }*/ srand(arg); num = rand()&0xFFFFFF; //pthread_mutex_unlock(&mutex);   printf("thread num %lx\n",tid);   ntries = 0; pthread_setcancelstate(PTHREAD_CANCEL_ENABLE,NULL); pthread_setcanceltype(PTHREAD_CANCEL_DEFERRED,NULL);   pthread_cleanup_push((void *)print_it,(void *)&ntries);   while(1) { num = (num+1)&0xffffff; ntries++;   if(arg == num) { //只允许一个线程操作此处 while(pthread_mutex_trylock(&mutex) == EBUSY) { //一个线程操作后其余线程进入次循环挂起,等待pthread_cancel函数发送cancel信号终止线程 k++; if(k == 10000) { printf("----------2busy2-----------\n"); }   pthread_testcancel(); } tries = ntries; //pthread_mutex_unlock(&mutex);  //如果加上这句话,将会有好几个线程找到主函数中设定的值pid printf("Thread %lx found the number!\n",tid);   for(j = 0;j<THREAD_MAX;j++) { if(thread[j]!=tid) { pthread_cancel(thread[j]); } }   break; } if(ntries%100 == 0) { h++; /*线程阻塞,其他线程争夺资源,或者是等待pthread_cancel函数发送cancel信号终止线程*/ pthread_testcancel(); /*这是为了弄明白pthread_testcancel函数的作用而设置的代码段*/ if(h == 10000) { h = 0; printf("----------thread num %lx-------------\n",tid); } } } pthread_cleanup_pop(0); return (void *)0; }     int main() { int i,pid;   pid = getpid(); //设置要查找的数   pthread_mutex_init(&mutex,NULL); printf("Search the num of %d\n",pid); for(started = 0; started < THREAD_MAX; started++) { pthread_create(&thread[started],NULL,(void *)Search_Num,(void *)pid); }   for(i = 0; i < THREAD_MAX; i++) { printf("-----------i = %d--------------\n",i); pthread_join(thread[i],NULL); } printf("It took %d tries ot find the number!\n",tries); return 0; }

运行结果:

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 Search the num of 6531 -----------i = 0-------------- thread num b6fbcb70 thread num b67bbb70 thread num b5fbab70 thread num b77bdb70 ----------thread num b67bbb70------------- Thread b67bbb70 found the number! ----------thread num b6fbcb70------------- ----------thread num b77bdb70------------- ----------2busy2----------- ----------thread num b5fbab70------------- ----------2busy2----------- Thread b5fbab70 was canceled on its 1174527 try. Thread b77bdb70 was canceled on its 1023100 try. -----------i = 1-------------- Thread b6fbcb70 was canceled on its 1174527 try. -----------i = 2-------------- -----------i = 3-------------- It took 1174527 tries ot find the number!

从这结果里你有没有看出什么呢?

 

未经允许不得转载:搬瓦工中文网 » Linux下高效管理:详解线程取消与终止的实用技巧