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

例子详细解析:

一. 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中两个指针,也就是说,这两个指针所存储的地址已经被两个线程的返回值所覆盖。

 

 

 

  

 

 

 

 

 

 

 

 

 

 

 

 

posted @ 2012-08-22 16:19  blankqdb  阅读(19347)  评论(0编辑  收藏  举报