线程创建
函数原型及头文件
#include <pthread.h>
int pthread_create(pthread_t *restrict tidp, const pthread_attr_t *restrict attr, void *(*start_rtn)(void *), void *restrict arg);
参数解读
tidp | 当pthread_create成功返回时,由tidp指向的内存单元被设置为新创建线程的线程ID。(指针,输入方式是地址) |
attr | 一般设为NULL。attr参数用于定制各种不同的线程属性,暂可以把它设置为NULL,以创建默认属性的线程。 |
(*start_rtn)(void *) | 新创建的线程的入口地址。新创建的线程从start_rtn函数的地址开始运行,该函数只有一个无类型指针参数arg。 |
arg | 向start_rtn函数传递的参数。如果需要向start_rtn函数传递的参数不止一个,那么需要把这些参数放到一个结构中,然后把这个结构的地址作为arg参数传入。(指针,输入方式是地址,同时要注意是void *型) |
返回值
若成功返回0,否则返回错误编号。
线程退出
单个线程可以通过以下三种方式退出,在不终止整个进程的情况下停止它的控制流:
1)线程只是从启动例程中返回,返回值是线程的退出码。
2)线程可以被同一进程中的其他线程取消。
3)线程调用pthread_exit。
函数原型及头文件
#include <pthread.h>
int pthread_exit(void *rval_ptr);
参数解读
rval_ptr:一个无类型指针,与传给启动例程的单个参数类似。进程中的其他线程可以通过调用pthread_join函数访问到这个指针。
线程等待
调用这个函数的线程将一直阻塞,直到指定的线程调用pthread_exit、从启动例程中返回或者被取消。如果例程只是从它的启动例程返回i,rval_ptr将包含返回码。如果线程被取消,由rval_ptr指定的内存单元就置为PTHREAD_CANCELED。
可以通过调用pthread_join自动把线程置于分离状态,这样资源就可以恢复。如果线程已经处于分离状态,pthread_join调用就会失败,返回EINVAL。
函数原型及头文件
#include <pthread.h>
int pthread_join(pthread_t thread, void **rval_ptr);
参数解读
thread | 线程名 |
**rval_ptr | 线程的返回值。如果对线程的返回值不感兴趣,可以把rval_ptr置为NULL。在这种情况下,调用pthread_join函数将等待指定的线程终止,但并不获得线程的终止状态。 |
返回值
若成功返回0,否则返回错误编号。
线程脱离
函数原型及头文件
#include <pthread.h>
int pthread_detach(pthread_t thread);
参数解读
一个线程或者是可汇合(joinable,默认值),或者是脱离的(detached)。当一个可汇合的线程终止时,它的线程ID和退出状态将留存到另一个线程对它调用pthread_join。脱离的线程却像守护进程,当它们终止时,所有相关的资源都被释放,我们不能等待它们终止。如果一个线程需要知道另一线程什么时候终止,那就最好保持第二个线程的可汇合状态。
pthread_detach函数把指定的线程转变为脱离状态。
返回值
若成功返回0,否则返回错误编号。
其他
本函数通常由想让自己脱离的线程使用,就如以下语句:
pthread_detach(pthread_self());
线程ID获取
函数原型及头文件
#include <pthread.h>
pthread_t pthread_self(void);
返回值
调用线程的ID。
线程比较
对于线程ID比较,为了可移植操作,我们不能简单地把线程ID当作整数来处理,因为不同系统对线程ID的定义可能不一样。
函数原型及头文件
#include <pthread.h>
int pthread_equal(pthread_t tid1, pthread_t tid2);
参数解读
tid1 | 线程1的名字 |
tid2 | 线程1的名字 |
返回值
若相等则返回非0值,否则返回0。
代码示例
线程返回数字打印
#include <stdio.h>
#include <pthread.h>void *func1(void *arg)
{static int ret = 1;//static保留数据 不然函数调用后空间无效printf("t1:this thread t1:%ld \n",(unsigned long)pthread_self());//self是void型 需对其进行强转printf("t1:num=%d\n",*((int *)arg));//传参过来的是主函数的num num取地址传过来需先将void型转成int型 再通过指针指向其地址得到数据pthread_exit((void *)&ret);//线程的退出 可附带线程的返回值 可通过pthread_join调用得到该数据
}
int main()
{int ret;int num =999;int *pret;pthread_t t1;ret = pthread_create(&t1,NULL,func1,(void *)&num);//第四个参数是void *型 而num是int型 需要对其进行强转 同时其输出为指针 则需要加上地址符号if(ret == 0)//判断线程是否创建成功{printf("main:create thread success\n");}printf("main:%ld\n",(unsigned long)pthread_self()); pthread_join(t1,(void **)&pret);//线程等待 等待上面函数退出之后再执行下面语句printf("main:ti quit %d\n",*pret);//打印出线程的返回值return 0;
}
线程返回字符串打印
#include <stdio.h>
#include <pthread.h>void *func1(void *arg)
{static char *p = "hello word";printf("t1:this thread t1:%ld \n",(unsigned long)pthread_self());printf("t1:num=%d\n",*((int *)arg));pthread_exit((void *)p);
}
int main()
{int ret;int num =999;char *pret;pthread_t t1;ret = pthread_create(&t1,NULL,func1,(void *)&num);if(ret == 0){printf("main:create thread success\n");}printf("main:%ld\n",(unsigned long)pthread_self()); pthread_join(t1,(void **)&pret);printf("main:ti quit %s\n",pret);//字符串的名字为地址 不需要加地址符return 0;
}