目录
线程特点
通常线程指的是共享相同地址空间的多个任务
使用线程的好处:大大提高了任务切换的效率
避免了额外的TLB&cache的刷新
线程共享资源
一般进程中的多个线程共享以下资源
可执行命令
静态数据
进程中打开的文件描述符
当前工作目录
用户ID
用户组ID
线程私有资源
每个线程私有资源包括:
线程ID(TID)
PC(程序计数器)和相关寄存器
堆栈
错误号(errno)
优先级
执行状态和属性
Linux线程库
pthread现场库中提供了如下基本操作
创建线程
回收线程
结束线程
同步和互斥机制
信号量
互斥锁
创建线程函数:pthread_create
#include <pthread.h>\
int pthread create(pthread_t *thread,const pthread_attr_t *attr, (void *)(*routine)(void*),void *arg);
成功时返回0,失败返回错误码
thread线程对象
attr线程属性,NULL代表默认属性
routine线程可执行函数
arg传递给routine的参数,参数是void *,注意传递参数格式
pthread_t pthread_self(void)查看自己的TID
注意:线程创建需要时间,如果主线程马上退出,线程不能得到执行(主进程退出,它创建的线程也会退出)
编译错误分析
1.createP_t.c:14:36: warning: passing argument 3 of ‘pthread_create’ from incompatible pointertype [-Wincompatible-pointer-types]
ret = pthread_create(&tid,NULL,testThread,NULL);
In file included from createP_t.c:1:0:
/usr/include/pthread.h:233:12: note: expected ‘void * (*)(void *)’ but argument is of type ‘int *(*)(char *)’
意义:表示pthread_create参数3的定义和实际代码不符合,期望的是void * (*)(void *),实际的代码是int * (*)(char *)
解决方法:改为pthread_create(&tid,NULL,(void*)testThread,NULL);
2.createP_t.c:(.text+0x4b):对‘pthread_create’未定义的引用
collect2: error: ld returned 1 exit status--------这个链接错误,
表示pthread_create这个函数没有实现
解决方法:编译时候加-lpthread
线程结束函数-pthread_exit
#include <pthread.h>
void pthread_exit(void *retval);
结束当前线程
retval可被其他线程通过pthread_join获取
线程私有资源被释放
线程查看tid函数-pthread_self
-pthread_t pthread_self(void) 查看自己的TID
-#include <pthread.h>
-pthread_t pthread_self(void);
获取线程的ID有两种方式:第一种通过pthread_create函数的第一个参数;通过在线程里面调用pthread_self函数
线程的参数传递(重点难点)
1、通过地址传递参数,注意类型转换
2、值传递,这时候编译器会报警,需要程序员自己保证数据长度准确
编译错误:
createP_t.c:8:34: warning: dereferencing ‘void *’ pointer
printf("input arg=%d\n",(int)*arg);
createP_t.c:8:5: error: invalid use of void expression
printf("input arg=%d\n",(int)*arg);
错误原因是void *类型指针不能直接用*取值(*arg),因为编译不知道数据类型。解决方法:转换为指定的指针类型后再用*取值比如:*(int *)arg
运行错误:*** stack smashing detected ***: ./mthread_t terminated
已放弃(核心已转储)
原因:栈被破坏了(数组越界)