Linux多线程编程并传递多个参数实例

来源:互联网 发布:如何入驻淘宝达人 编辑:程序博客网 时间:2024/06/05 19:10

转自: http://www.cnblogs.com/blankqdb/articles/2651029.html


例子详细解析:

一. pthread_create()与pthread_join()函数

1 #include <pthread.h>2 int pthread_join(pthread_t thread,3                         void **retval);

 

1. pthread_join函数作用

   pthread_join函数作用是在一个线程中以阻塞的方式等待另一个线程(线程标识符为thread)的退出。如果等待的进程已经结束,那么该函数会立即返回。

    retval是用户定义的指针,用来存储被等待线程的返回值

    返回值: 0 -- 成功,失败 -- 错误号errno

2. pthread_join的应用

    使一个线程等待另一个线程的结束

    代码中如果没有pthread_join主线程会很快结束,从而从而合整个进程线束,从而使创建的线程没有机会执行就结束了,在主线程加入pthread_join后,主线程会阻塞等待直到(被等待的)线程结束后,主线程自己才结束,从而使创建的线程有机会执行

3. 一个线程不能被多个线程等待,否则第一个接收到信号的线程成功返回,其余调用 pthread_join 的线程则返回错误代码ESRCH。

 

1 #include <pthread.h>2 int pthread_create(pthread_t *thread, 3                             const pthread_attr_t *attr,4                             void *(*start_routine) (void *), 5                             void *arg);

1. pthread_create函数的作用

   创建一个线程,成功时返回0,错误时返回errno。

   thread:被创建线程的标识符,pthread_join使用这个标识符来等待该线程的结束。

   attr:     设置线程的属性,可以为NULL

   第三个参数是线程函数的入口地址

   arg:      传递给线程的参数,当要传递给线程的参数有多个时,可以使用结构体.

复制代码
#ifdef HAVE_CONFIG_H#include <config.h>#endif#include <stdio.h>#include <stdlib.h>#include <pthread.h>#include <string.h>// 参数结构体 struct argument{  int num;  char string[30];};// 声明两个线程函数void *thread1_func( void * );void *thread2_func( void * );int main(int argc, char *argv[]){  //定义两个线程标识符  pthread_t thread1, thread2;  //定义用来接收两个线程退出后的返回值,用作pthread_join的第二个参数  void *thread1_return, *thread2_return;  //传递的参数结构体  struct argument arg1, arg2;  int i;  int wait_thread_end; //判断线程退出成功与否  //参数结构体值初始化  arg1.num = 1949;  strcpy( arg1.string, "中华人民共和国" );    arg2.num = 2012;  strcpy( arg2.string, "建国63周年" );    // 创建两个线程  pthread_create(&thread1, NULL, thread1_func, (void*)&arg1 );  pthread_create( &thread2, NULL, thread2_func, (void*)&arg2 );    for( i = 0; i < 2; i++ )  {    printf("我是最初的进程!\n");    sleep(2);  //主统线程睡眠,调用其他线程  }    //等待第一个线程退出,并接收它的返回值(返回值存储在thread1_return)  wait_thread_end = pthread_join( thread1, &thread1_return );  if( wait_thread_end != 0 )   {    printf("调用 pthread_join 获取线程1的返回值出现错误!\n");  }  else  {    printf("调用 pthread_join 成功!线程1退出后的返回值是 %d\n", (int)thread1_return);  }    //等待第二个线程退出,并接收它的返回值(返回值存储在thread2_return)  wait_thread_end = pthread_join( thread2, &thread2_return);  if( wait_thread_end != 0 )   {    printf("调用 pthread_join 获取线程2的返回值出现错误!\n");  }  else  {    printf("调用 pthread_join 成功!线程2退出后的返回值是 %d\n",(int)thread2_return );  }  return EXIT_SUCCESS;}/** *线程1函数实现  */ void *thread1_func( void *arg ){  int i;  struct argument *arg_thread1; // 接收传递过来的参数结构体    arg_thread1 = ( struct argument * )arg;    for( i = 0; i < 3; i++)  {    printf( "我来自线程1,传递给我的参数是 %d, %s\n", arg_thread1->num, arg_thread1->string);    sleep(2); // 投入睡眠,调用其它线程  }  return (void *)123;} void *thread2_func( void *arg ){  int i;  struct argument *arg_thread2; // 接收传递过来的参数结构体    arg_thread2 = ( struct argument * )arg;    for( i = 0; i < 3; i++)  {    printf( "我来自线程2,传递给我的参数是 %d, %s\n", arg_thread2->num, arg_thread2->string);    sleep(2); // 投入睡眠,调用其它线程  }  return (void *)456;}
复制代码

例子中要请注意的地方:

 void *thread1_func( void *arg )

  {

    ...

    return (void*)123;

  }

   void *thread2_func( void *arg )

  {

    ...

    return (void*)456;

  }

1. 在线程函数中thread1_func()和thread2_func()中,最后一句return语句中,对返回的值要进行类型转换(转换成(void *)),返回值的类型要与线程函数的声明和定义的返回值类型一致。

2.  两个线程函数的返回值均为一个指针(把一个整数转换成(void*)返回)。该指针存储在pthread_join()的第三个参数中,在这两个函数中等价于

    &thread1_return = thread1_func((void*)&arg1);

   &thread2_return = thread2_func((void*)&arg2);

   其中,&thread1_return是pthread_join的第二个参数,在前面函数解析中说过,pthread_join(phtread_t thread, void **retval)函数的第二个参数retval可以存储线程的返回值。该返回值直接存储在&thread1_return和&thread2_return,此时thread1_return和thread2_return值就是线程1和线程2函数的返回值(void *)类型。所以(int)thread1_return和(int)thread2_return就是该函数的返回值内容。由定义(void *thread1_return, *thread2_return)可以看出,thread1_return和thread2_return中两个指针,也就是说,这两个指针所存储的地址已经被两个线程的返回值所覆盖。


0 0
原创粉丝点击