Linux应用—多线程编程
多线程的使用
线程的概念
操作系统所能调度的最小单位。一个进程可以包含多个线程,每个线程共享进程资源
线程的标识
线程号(tid)和进程号(PID)是表示线程和进程的唯一标识
线程号本质上就是一个 pthread_t 结构体,其仅仅在其所属进程的上下文中才有意义
获取当前线程的 tid:
#include <pthread.h>
pthread_t pthread_self(void);
线程的创建
#include <pthread.h>
int pthread_create(pthread_t *thread, const pthread_attr_t *attr,
void *(*start_routine) (void *), void *arg);
—— thread
创建线程的线程 tid 号
—— attr
线程的属性一般设置为 BULL,表示默认属性
—— 第三个参数
线程函数的函数指针
—— arg
传给线程函数的参数,若没有参数可传,填 NULL
void *fun1(void *arg)
{
...
}
pthread_t tid1;
pthread_create(&tid1,NULL,fun,NULL);
线程传参
pthread_create()的最后一个参数的为void类型的数据,表示可以向线程传递一个 void数据类型的参数,线程的回调函数中可以获取该参数
传递的参数可以是变量也可以是变量的地址
#include <pthread.h>
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
void *fun1(void *arg)
{
printf("%s:arg = %d Addr = %p\n",__FUNCTION__,*(int *)arg,arg);
}
void *fun2(void *arg)
{
printf("%s:arg = %d Addr = %p\n",__FUNCTION__,(int)(long)arg,arg);
}
int main()
{
pthread_t tid1,tid2;
int a = 50;
int ret = pthread_create(&tid1,NULL,fun1,(void *)&a);
if(ret != 0){
perror("pthread_create");
return -1;
}
ret = pthread_create(&tid2,NULL,fun2,(void *)(long)a);
if(ret != 0){
perror("pthread_create");
return -1;
}
sleep(1);
printf("%s:a = %d Add = %p \n",__FUNCTION__,a,&a);
return 0;
}
线程的退出和回收
线程的退出有三种情况:
-
进程结束,所属该进程的所有线程也随之结束
-
线程调用 pthread_exit 函数主动退出线程
void pthread_exit(void *retval); // 线程可以传入参数给主线程
-
其他线程调用 pthread_cancel 函数使进程被动退出
int pthread_cancel(pthread_t thread); // 强制使该线程号指向的线程退出
线程的回收:
线程结束后,主线程通过函数 pthread_join/pthread_tryjoin_np 来回收线程的资源,并且获得线程结束后需要返回的数据
int pthread_join(pthread_t thread, void **retval);
该函数为线程回收函数,默认状态为阻塞状态,直到成功回收线程后才返回。第一个参数为要回收线程的 tid 号,第二个参数为线程回收后接受线程传出的数据。
int pthread_tryjoin_np(pthread_t thread, void **retval);
该函数为非阻塞模式回收函数,通过返回值判断是否回收掉线程,成功回收则返回 0,其余参数与 pthread_join 一致。
总结
线程的控制
互斥锁
读写锁
条件变量
信号量