用pthread传递参数格式一般是这样的:
int pthread_create (pthread_t *tid,const pthread_attr_t *attr,void *(*func)(void *),void *arg)
其中最后一个参数就是用来传递参数的,可以是一个简单的参数,也可以是一个参数集合成的结构体。
由于线程之间使用不同的栈,因此,如果是本地的局部变量作为参数传递到新线程中使用,有可能使参数的值变成未定义的,简单举例如下:
struct node_ {
int index;
string name;
}node;
void *thread_callback(void *arg){
node_ *pc = (node_*)arg;
int localIndex = pc->index;
string localName = pc->name;
.....
return NULL;
}
int test(string name){
node_ param;
param.index = 5;
param.name = name;
pthread_t pid; pthread_create(&pid, NULL, thread_callback, (void *)¶m);
......
return 0;
}
上面的例子中,把本地的参数param的地址传给了新线程,在新线程中会取出该参数使用,这就是把原来线程的局部变量传递到新线程中,在新线程中使用该变量时行为是未定义的,也就是index和name的值都是随机的,也有可能是无法访问的地址值,可能导致程序行为异常或者访问了非法地址而导致崩溃。
如何正确传递参数呢?由于堆是共享的,可以将栈数据转化为堆数据,把堆的地址传递到新的线程,这样能够正确的传递变量值。
int test(string name){
node_ *param = new node_;
param->index = 5;
param->name = name;
pthread_t pid; pthread_create(&pid, NULL, thread_callback, (void *)param);
......
return 0;
}
这样我们会发现,param的地址跟arg的地址是相同的,也就是把堆的一个地址作为参数传递到新线程,从而达到了共享数据的目的。
需要注意的是,新线程中取出数据后需要把指针删除,防止内存泄露。
也可以将局部变量转化为全局变量来传递数据。