pthread_join/pthread_exit用法实例

本文介绍了线程的创建、等待和退出机制,通过实例展示了如何使用pthread_join函数实现线程间的等待与资源回收,以及pthread_exit函数用于线程的正常退出。详细解析了线程生命周期中的关键操作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

函数pthread_join用来等待一个线程的结束。函数原型为:
  extern int pthread_join __P ((pthread_t __th, void **__thread_return));
  第一个参数为被等待的线程标识符,第二个参数为一个用户定义的指针,它可以用来存储被等待线程的返回值。这个函数是一个线程阻塞的函数,调用它的线程将一直等待到被等待的线程结束为止,当函数返回时,被等待线程的资源被收回。一个线程的结束有两种途径,一种是象我们上面的例子一样,函数结束了,调用它的线程也就结束了;另一种方式是通过函数pthread_exit来实现。它的函数原型为:
  extern void pthread_exit __P ((void *__retval)) __attribute__ ((__noreturn__));
  唯一的参数是函数的返回代码,只要pthread_exit中的参数retval不是NULL,这个值将被传递给 thread_return。最后要说明的是,一个线程不能被多个线程等待,否则第一个接收到信号的线程成功返回,其余调用pthread_join的线程则返回错误代码ESRCH。

实例:

/*myfile11-3.c*/


#include <stdio.h>
#include <pthread.h>
#include <stdlib.h>

pthread_t       tid1, tid2; 
void            *tret;

 

void *
thr_fn1(void *arg)
{
        sleep(1);//睡眠一秒,等待TID2结束。
        pthread_join(tid2, &tret);//tid1一直阻赛,等到tid2的退出,获得TID2的退出码
         printf("thread 2 exit code %d\n", (int)tret);
    printf("thread 1 returning\n");
    return((void *)2);
}

void *
thr_fn2(void *arg)
{      
    printf("thread 2 exiting\n");
     pthread_exit((void *)3);
}

int
main(void)
{
    int            err;

    err = pthread_create(&tid1, NULL, thr_fn1, NULL);
    if (err != 0)
        printf("can't create thread 1\n");
    err = pthread_create(&tid2, NULL, thr_fn2, NULL);
    if (err != 0)
        printf("can't create thread 2\n");
    err = pthread_join(tid1, &tret);//祝线程一直阻赛,等待TID1的返回。
    if (err != 0)
        printf("can't join with thread 1\n");
    printf("thread 1 exit code %d\n", (int)tret);
      //err = pthread_join(tid2, &tret);
    //if (err != 0)
    //    printf("can't join with thread 2\n");
//    printf("thread 2 exit code %d\n", (int)tret);
    exit(0);
}

 


命令:#gcc -lthread myfile11-3.c

        :#./a.out

运行结果:

thread 2 exiting
thread 2 exit code 3
thread 1 returning
thread 1 exit code 2

``` static int s_iServerSocket = 0; static stNetinfo s_stNetinfo[cmnDfn_WORKING_PTHREAD_NUMBER / TASK_THREADS] = { 0 }; int TCPSvr_Run(void) { int iRet = -1; int iClientSocket; int iTaskID = 0; int iCLientNumber = 0; struct sockaddr_in stClientaddr; socklen_t addrlen = sizeof(stClientaddr); pthread_t AcceptTrd; if (s_iServerSocket <= 0) { cmn_PrtError("server socket not available"); } if(listen(s_iServerSocket,cmnDfn_MAX_CLIENT) < 0) { cmn_PrtError("listen error"); } s_stNetinfo->iSocket = s_iServerSocket; if(pthread_create(&AcceptTrd,NULL,AcptTrd,s_stNetinfo) != 0) { cmn_PrtError("Error in creating thread"); } if(pthread_join(AcceptTrd,NULL) < 0) { cmn_PrtError("Error in joining thread"); } _Exit: return iRet; } void* AcptTrd(IN OUT void *arg) { stNetinfo *Netinfo = (stNetinfo *) arg; //int iListenSocket = (int *)arg; int iListenSocket = Netinfo->iSocket; int iClientSocket; int iTaskID = 0; int iLoopNum; //int iCLientNumber = 0; struct sockaddr_in stClientaddr; socklen_t addrlen = sizeof(stClientaddr); //stNetAddr stClientArray[cmnDfn_WORKING_PTHREAD_NUMBER / 2] = {0}; while (g_iExit == 0) { iClientSocket = accept(iListenSocket, (struct sockaddr *)&stClientaddr,&addrlen); if(iClientSocket < 0) { if (errno == EINTR) { if (g_iExit == 1) { break; } continue; } else { cmn_PrtError("accept error"); } } for (iLoopNum = 0; iLoopNum < cmnDfn_WORKING_PTHREAD_NUMBER / TASK_THREADS; iLoopNum++) { if (Netinfo[iLoopNum].iRcvIsAvailable == 0 && Netinfo[iLoopNum].iSndIsAvailable == 0) { pthread_mutex_init(&Netinfo[iLoopNum].stMutex, NULL); LOCK(Netinfo[iLoopNum].stMutex) { Netinfo[iLoopNum].iSocket = iClientSocket; Netinfo[iLoopNum].iClientNumber = iLoopNum+1; if(ThreadPoolSubmit(TCPRcvTrd, &Netinfo[iLoopNum],&iTaskID) < 0) { cmn_PrtError("Error in submitting task to thread pool"); } Netinfo[iLoopNum].iRcvIsAvailable = 1; if(ThreadPoolSubmit(TCPSndTrd, &Netinfo[iLoopNum],&iTaskID) < 0) { cmn_PrtError("Error in submitting task to thread pool"); } Netinfo[iLoopNum].iSndIsAvailable = 1; } break; } } if(iLoopNum >= cmnDfn_WORKING_PTHREAD_NUMBER / TASK_THREADS) { printf("\033[%d;0H\033[KNo more available client vacancies",iLoopNum); fflush(stdout); close(iClientSocket); continue; } } _Exit: pthread_exit(NULL); }```检查线程参数传入和在线程内修改参数是否问题存在
03-25
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值