11.2 线程概念
线程资源:线程ID,一组寄存器,栈,调度优先级和策略,信号屏蔽字,errno变量,线程私有数据。
进程的共享资源:进程的可执行代码,全局内存,堆内存,栈和文件描述符
11.3 线程标识
int pthread_equal(pthread_t tid1, pthread_t tid2);int pthread_self();
Linux使用无符号长整形表示pthread_t
11.4 线程创建
int pthread_create(pthread_t *restrict tidp, const pthread_attr_t *restrict attr,void *(start_rtn(void*)), void* restrict arg);
在调用失败后,返回错误码,而不是设置errno
11.5 线程终止
如果进程中任意线程调用了exit、_Exit或者_exit,那么整个进程就会终止。
单个线程可以通过3种方式退出,因此可以在不终止整个进程的情况下,停止它的控制流:
线程可以简单地从启动例程中返回,返回值是线程的退出码
线程可以被同一进程中的其他线程取消
线程调用pthread_exit
void pthread_exit(void *rval_ptr);
int pthread_join(pthread_t thread,void **rval_ptr);
调用pthread_join后,调用线程将一直阻塞,直到指定的线程退出。利用rval_ptr_ptr 访问 rval_ptr
int pthread_cancel(pthread_t tid);
线程可以通过调用pthread_cancel函数来请求取消同一进程中的其他进程。
pthread_cancel并不等待线程终止,它仅仅提出请求,线程可以选择
忽略取消或者控制如何被取消。
void pthread_cleanup_push(void (*rtn)(void *),void *arg);
void pthread_cleanup_pop(int execute);
1 .线程可以安排它退出时需要调用的函数,这与进程在退出时可以用atexit函数安排退出时类似的。
2. 如果线程是通过从它的启动例程中退出返回而终止的话,它的清理处理程序就不会被调用。
3. rtn函数并不是只有在线程结束时,才会被调用。参看第四条
4. 如果pthread_cleanup_pop被调用且execute参数设置为非0,则[调用并删除]上次pthread_cleanup_push调用建立的清理处理程序。如果execute参数为0,则清理函数将不被调用(只删除)。
进程线程函数的相似之处
第12章 线程属性
12.2 线程限制
12.3 线程属性
线程的分离状态属性
PTHREAD_CREATE_DETACHED
PTHREAD_CREATE_JOINABLE (默认)
线程栈的最低地址
线程栈的大小
线程栈末尾的警戒缓冲区大小
可取消状态
PTHREAD_CANCEL_ENABLE(默认)
PTHREAD_CANCEL_DISABLE
可取消类型
PTHREAD_CANCEL_DEFERRED(默认)
PTHREAD_CANCEL_ASYNCHRONOUS
int pthread_attr_setdetachstate(pthread_attr_t *attr, int detachstate);
int pthread_attr_getdetachstate(pthread_attr_t *attr, int *detachstate);
如果不想了解现有某个线程的终止状态,可以使用pthread_detach函数让操作系统在线程退出时候收回它所占用的资源。
创建线程时候可以修改pthread_attr_t结构中的detachstate线程属性,让线程以分离状态启动。
detatchstate值:
PTHREAD_CREATE_DETACHED: 分离状态启动
PTHREAD_CREATE_JOINABLE 正常启动,应用程序可以获取线程的终止状态
int pthread_attr_setstack(pthread_attr_t *attr,void *stackaddr, size_t stacksize);
int pthread_attr_getstack(pthread_attr_t *attr,void **stackaddr, size_t *stacksize);
尽量使用 pthread_attr_setstacksize, 除非线程对栈的地址空间有特殊要求
int pthread_attr_setstacksize(pthread_attr_t *attr, size_t stacksize);
int pthread_attr_getstacksize(pthread_attr_t *attr, size_t *stacksize);
linux3.2.0 线程的默认栈空间为8MB,可以设定的最小栈空间为16KB
int pthread_attr_setguardsize(pthread_attr_t *attr, size_t guardsize);
int pthread_attr_getguardsize(pthread_attr_t *attr, size_t *guardsize);
1. guardsize控制着线程栈末尾之后用以避免栈溢出的扩展内存的大小。当栈指针溢出到警戒区时,应用程序接受到出错信息
2. 若利用pthread_attr_setstack,系统认为我们将自己管理栈,进而使栈警戒缓冲区机制无效,相当于guardsize等于0
3. linux guardsize的默认大小为4KB,
int pthread_setcancelstate(int stat,int *oldstate);
若成功,返回0;否则,返回错误编号
1.调用该函数且stat为PTHREAD_CANCEL_DISABLE后,取消请求处于挂起状态。当取消状态再次变为PTHREAD_CANCLE_ENABLE时,
线程将在下一个取消点上对所有的取消请求进行处理。
2.pthread_cancel(tid)调用并不等待线程终止。在默认情况下,线程tid在取消请求发出以后还是继续运行,直到线程达到某个取消点。
void pthread_testcancel(void);
可以调用pthread_testcancel函数在程序中添加自己的取消点
int pthread_setcanceltype(int type,int *oldtype);
type参数可以是PTHREAD_CANCEL_DEFERRED(默认),也可以是PTHREAD_CANCEL_ASYNCHRONOUS(异步取消)。
如果使用异步取消。线程可以在任意时间撤销(linux上一般接收到信号便结束),而不是遇到取消点才能被取消。