pthread_join函数

来源:互联网 发布:深圳现暴力接亲 知乎 编辑:程序博客网 时间:2024/03/29 07:44

原文地址:点击打开链接

函数原型:int pthread_join(pthread_t tid, void **status);


功能:pthread_join()函数会一直阻塞调用线程,直到指定的线程tid终止。当pthread_join()返回之后,应用程序可回收

与已终止线程关联的任何数据存储空间,(另外也可设置线程attr属性,当线程结束时直接回收资源)如果没有必要等待特定的线程

终止之后才进行其他处理,则应当将该线程分离pthread_detach()。


头文件:#include <pthread.h>

pthread非linux系统的默认库, 需手动链接-线程库 -lpthread


参数:

tid:需要等待的线程,指定的线程必须位于当前的进程中,而且不得是分离线程

status:线程tid所执行的函数start_routine的返回值(start_routine返回值地址需要保证有效),其中status可以为nullptr


返回值:

调用成功完成后,pthrea_join()返回0.其他任何返回值都表示出现了错误。如果检测到以下任意情况

pthread_join()将失败并返回相应的值

ESRCH
描述: 没有找到与给定的线程ID 相对应的线程。(如果多个线程等待同一个线程终止,则所有等待线程将一直等到目标线程终止。

然后一个等待线程成功返回。其余的等待线程将失败返回ESRCH错误)
EDEADLK
描述: 将出现死锁,如一个线程等待其本身,或者线程A和线程B 互相等待。
EINVAL
描述: 与给定的线程ID 相对应的线程是分离线程。


代码:

[cpp] view plain copy
  1. #include <iostream>  
  2. #include <pthread.h>  
  3. #include <cstdlib>  
  4. #include <string>  
  5. #include <unistd.h>  
  6. using std::cout;  
  7. using std::cin;  
  8. using std::cerr;  
  9. using std::endl;  
  10. using std::string;  
  11. struct info  
  12. {  
  13.     string name;  
  14.     unsigned s_time;//sleep时间  
  15. };  
  16. void *start_routine(void *arg)  
  17. {  
  18.     info *t_arg = (info *) arg;  
  19.     for(unsigned i = 0; i != 5U; ++i)  
  20.     {  
  21.         cout << t_arg->name << endl;  
  22.         sleep(t_arg->s_time);  
  23.     }  
  24.     return (void *)t_arg;  
  25. }  
  26. constexpr unsigned THREAD_N = 2;//线程的数量  
  27. int main()  
  28. {  
  29.     pthread_t tid[THREAD_N];  
  30.     //pthread_attr_t *attr1 = nullptr, *attr2 = nullptr;  
  31.     int ret[THREAD_N];  
  32.     info args[THREAD_N] = {"Win", 1U, "Unix", 2U};  
  33.     for(unsigned i = 0; i != THREAD_N; ++i)  
  34.     {  
  35.         /* 
  36.            pthread_attr_t attr; 
  37.            ret[i] = pthread_create(&tid[i], &attr, &start_routine, &args[i]); 
  38.            attr需由pthread_attr_init()进行初始化,此处产生了EAGAIN错误 
  39.          */  
  40.         ret[i] = pthread_create(&tid[i], nullptr, &start_routine, &args[i]);  
  41.         if(ret[i] != 0)  
  42.         {  
  43.             if(ret[i] == EAGAIN)  
  44.             {  
  45.                 cerr << args[i].name << " : 超出系统限制" << endl;  
  46.             }     
  47.             else if(ret[i] == EINVAL)  
  48.             {  
  49.                 cerr << args[i].name << " : pthread_attr_t设置无效" << endl;  
  50.             }  
  51.             else   
  52.             {  
  53.                 cerr << args[i].name << ": unkown error" << endl;  
  54.             }     
  55.             exit(-1);  
  56.         }  
  57.     }  
  58.     /* 
  59.         对于pthread_join(),如果线程tid的start_routine已经执行完毕, 
  60.         pthrea_join仍能正常返回,这说明start_routine执行完毕后线程 
  61.         资源并没有被收回 
  62.     */  
  63.     void *status[THREAD_N];  
  64.     for(unsigned i = 0; i != THREAD_N; ++i)  
  65.     {  
  66.         ret[i] = pthread_join(tid[i], &status[i]);  
  67.         if(ret[i] != 0)  
  68.         {  
  69.             if(ret[i] == ESRCH)  
  70.             {  
  71.                 cerr << "pthread_join():ESRCH 没有找到与给定线程ID相对应的线程" << endl;  
  72.             }  
  73.             else if(ret[i] == EDEADLK)  
  74.             {  
  75.                 cerr << "pthread_join():EDEADLKI 产生死锁" << endl;  
  76.             }  
  77.             else if(ret[i] == EINVAL)  
  78.             {  
  79.                 cerr << "pthread_join():EINVAL 与给定的县城ID相对应的线程是分离线程" << endl;  
  80.             }  
  81.             else  
  82.             {  
  83.                 cerr << "pthread_join():unknow error" << endl;  
  84.             }  
  85.             exit(-1);  
  86.         }  
  87.     }  
  88.     /* 
  89.         ret[0] = pthread_join(tid[0], &status[0]); 
  90.         cerr << (ret[0] == ESRCH); 
  91.         由于线程tid[0]的资源已被收回,所以此处产生ESRCH 
  92.     */  
  93.     return 0;  

原创粉丝点击