三、linux中pthread_join()与pthread_detach()解析

关于线程用法可以参考这些文章
一、Linux中 C/C++线程使用

二、Pthread 锁与 C++读写锁

三、linux中pthread_join()与pthread_detach()解析

四、linux中pthread_cond_wait()与pthread_cond_signal ()解析

Note: 关于内核使用线程方法可以参考之前写的另外一篇文章

内核线程(kthread)的简单使用

这篇文章主要是介绍 pthread两种状态: joinable状态和unjoinable状态

  1. linux线程执行 pthread有两种状态joinable状态和unjoinable状态
    默认是joinable 状态,可以通过这个api 来获取其状态 pthread_attr_getdetachstate
    另外可以通过如下代码来设置为状态joinable 还是 unjoinable
pthread_attr_t attr;
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
pthread_create(&thr, &attr, &thread_start, NULL);

更多细节可以看 pthread_create , pthread_attr_init

还有一种方法是线程创建后在线程中调用 pthread_detach, 如:pthread_detach(pthread_self()),将状态改为unjoinable状态,确保资源的释放。

void  ThreadFunc( void *ptr )
{       
        pthread_detach(pthread_self());
        pthread_exit(0) ;
}

 pthread_t tid;
 int status = pthread_create(&tid, NULL, ThreadFunc, NULL);
 if(status != 0)
 {
     perror("pthread_create error");
 }

或者外部主线程主动调用 pthread_detach(tid)

 pthread_t tid;
 int status = pthread_create(&tid, NULL, ThreadFunc, NULL);
 if(status != 0)
 {
     perror("pthread_create error");
 }
 pthread_detach(tid);
  1. 如果线程是joinable状态,当线程函数自己返回退出时或pthread_exit时都不会释放线程所占用堆栈和线程描述符(总计8K多)。只有当你调用了pthread_join之后这些资源才会被释放。
  2. 若是unjoinable状态的线程,堆栈和线程描述符这些资源在线程函数退出时或pthread_exit时自动会被释放。

总的来说就是在线程属性设置为unjoinable,在函数尾部直接 pthread_exit线程就会自动退出,省去了给线程主动调用pthread_join 回收资源的麻烦。

Demo 1:使用默认状态

#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
void *thread_function(void *arg)
{
    int i;
    for (i = 0; i < 8; i++)
    {
        printf("%s:Thread working...! %d \n", __FUNCTION__, i);
        sleep(1);
    }
    return NULL;
}

int main(void)
{
    pthread_t mythread;

    if (pthread_create(&mythread, NULL, thread_function, NULL))
    {
        printf("error creating thread.");
        abort();
    }
    /*
    if ( pthread_join ( mythread, NULL ) )
    {
        printf("error join thread.");
        abort();
    }
    */
    printf("%s:Thread done! \n", __FUNCTION__);
    exit(0);
}

打印输出:

main:Thread done!

因为主线程main 跑的比子线程快,子线程中 thread_function 的打印是不是打印出来的

Demo 2:使用pthread_join 阻塞主线程

(1)pthread_join()即是子线程合入主线程,主线程阻塞等待子线程结束,然后回收子线程资源。
(2)函数说明
1)头文件 : #include <pthread.h>
2)函数定义: int pthread_join(pthread_t thread, void **retval);
3)描述 :pthread_join()函数,以阻塞的方式等待thread指定的线程结束。当函数返回时,被等待线程的资源被收回。如果线程已经结束,那么该函数会立即返回。并且thread指定的线程必须是joinable的。
4)参数 :thread: 线程标识符,即线程ID,标识唯一线程。retval: 用户定义的指针,用来存储被等待线程的返回值。
5)返回值 : 0代表成功。 失败,返回的则是错误号。
将上述Demo 1 中的注释放开,可以看到 子线程中 thread_function 的打印

#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
void *thread_function(void *arg)
{
    int i;
    for (i = 0; i < 8; i++)
    {
        printf("%s:Thread working...! %d \n", __FUNCTION__, i);
        sleep(1);
    }
    return NULL;
}

int main(void)
{
    pthread_t mythread;

    if (pthread_create(&mythread, NULL, thread_function, NULL))
    {
        printf("error creating thread.");
        abort();
    }
    /*
    if ( pthread_join ( mythread, NULL ) )
    {
        printf("error join thread.");
        abort();
    }
    */
    printf("%s:Thread done! \n", __FUNCTION__);
    exit(0);
}

打印输出

thread_function:Thread working...! 0
thread_function:Thread working...! 1
thread_function:Thread working...! 2
thread_function:Thread working...! 3
thread_function:Thread working...! 4
thread_function:Thread working...! 5
thread_function:Thread working...! 6
thread_function:Thread working...! 7
main:Thread done!

Demo 3:使用pthread_detach()

(1)pthread_detach()即主线程与子线程分离,子线程结束后,资源自动回收。
(2)函数说明
1)函数原型:int pthread_detach(pthread_t tid);
2)功能:pthread_join()函数的替代函数,可回收创建时detachstate属性设置为PTHREAD_CREATE_JOINABLE的线程的存储空间。该函数不会阻塞父线程。pthread_join()函数用于只是应用程序在线程tid终止时回收其存储空间。如果tid尚未终止,pthread_detach()不会终止该线程。当然pthread_detach(pthread_self())也是可以得
3)头文件:#include <pthread.h> pthread非linux系统的默认库, 需手动链接-线程库 -lpthread
4)参数:tid:线程标识符
5)返回值:pthread_detach() 在调用成功完成之后返回零。其他任何返回值都表示出现了错误。如果检测到以下任一情况,pthread_detach()将失败并返回相应的值。
EINVAL:tid是分离线程
ESRCH:tid不是当前进程中有效的为分离线程

#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
void *thread_function(void *arg)
{
    pthread_detach(pthread_self());
    int i;
    for (i = 0; i < 8; i++)
    {
        printf("%s:Thread working...%d \n", __FUNCTION__, i);
        sleep(1);
    }
    pthread_exit(NULL);
}

int main(void)
{
    pthread_t mythread;

    if (pthread_create(&mythread, NULL, thread_function, NULL))
    {
        printf("error creating thread.");
        abort();
    }
    sleep(5);
    printf("%s: tid:%lu\n", __FUNCTION__, mythread);
    printf("%s: Thread work done! \n", __FUNCTION__);
    exit(0);
}

打印输出

thread_function:Thread working...0
thread_function:Thread working...1
thread_function:Thread working...2
thread_function:Thread working...3
thread_function:Thread working...4
main: tid:1
main: Thread work done!
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342