4.1.3 线程返回值
如果你传递给pthread_join的第二个参数不为NULL,那么线程的返回值将会被放在这个参数指向的位置。线程的返回值为
void*类型。在进行数据类型转换时你应当确保没有位丢失。
#include <pthread.h>
#include <stdio.h>
/*找到第N个素数*/
void* compute_prime (void* arg)
{
int candidate = 2;
int n = *((int*) arg);
while (1) {
int factor;
int is_prime = 1;
/*测试是不是素数*/
for (factor = 2; factor < candidate; ++factor)
if (candidate % factor == 0) {
is_prime = 0;
break;
}
/*看是不是我们要找的素数*/
if (is_prime){
if ( --n == 0)
return (void*) candidate;
}
++candidate;
}
return NULL;
}
int main()
{
pthread_t thread;
int which_prime = 5000;
int prime;
pthread_create (&thread, NULL, &compute_prime, &which_prime);
pthread_join (thread, (void*)&prime);
printf("The %dth prime number is %d. /n", which_prime, prime);
}
4.1.4 线程ID的扩展
有时候我们需要知道那个线程在执行当前的代码。 pthread_self 函数可以返回本线程的ID,pthread_equal 可以用来比较两个线程
ID是否相同。如下的代码可以避免自己等待自己而造成死锁(实际上pthread_join 会返回EDEADLK)
if(!pthread_equial (pthread_self (), other_thread))
pthread_join(other_thread,NULL);
4.1.5 线程属性
线程属性提供了一种调节单独线程的行为的机制。pthread_cteate的第二个参数与线程的属性相关。如果简单的传入NULL那么将使用
默认值创建线程。要设定一个线程的属性可以进行如下的步骤:
1.创建一个pthread_attr_t对象。
2.调用pthread_attr_init传递第一步中的pthread_attr中地址。这将会把它初始化为默认的值。
3. 修改属性对象中的某些值为你想要的设置。
4. 调用pthread_create时传入这个属性变量的指针。
5.调用pthread_attr_destroy 来清除属性变量的值。pthread_attr_t 变量自身没有被释放掉,它可以通过pthread_attr_init再次使用。
一个单独的线程属性对象可以用来启动多个线程。而当线程启动后这个属性变量就不再需要。
对于大部分应用程序编程任务,有一个属性你可能比较感兴趣。那就是线程的分离状态(detach state)。默认情况下线程被创建为链接
线程(joinable thread),你可以设定为分离线程(detached thread)。类似于进程,一个链接线程结束后不是自动清理自身的线程信息。
一个线程的退出状态留在系统中直到另一个线程调用了pthread_join 获得了它返回码。只有这时它的资源才被释放。而分离线程(detached thread)结束后自动清理资源,其他线程不能使用pthread_join得到它的返回值。
使用pthread_attr_setdetachstate可以设定线程的分离状态属性。如下是个例子,
#include <pthread.h>
#include <stdio.h>
void* thread_fun(void* arg)
{
return (void*)123;
}
int main(void)
{
pthread_attr_t attr;
pthread_t thrd_id;
int ret_val = -1;
pthread_attr_init (&attr);
// pthread_attr_setdetachstate (&attr,PTHREAD_CREATE_DETACHED);//观察注释前后结果有什么不同
pthread_attr_destroy (&attr);
pthread_create (&thrd_id,&attr,&thread_fun,NULL);
pthread_join(thrd_id,(void*)&ret_val);
printf("/tThe return value is %d/n",ret_val);
return 0;
}
即使一个线程被创建为链接类型(joinable state)的,这可以改变为非链接类型的通过调用pthread_detach(),一旦一个线程设定为
detached类型它将不能再改为链接类型。