UNIX高级环境编程 第11、12章 线程及其属性

第11章 线程

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,则清理函数将不被调用(只删除)。

进程线程函数的相似之处

clipboard.png

第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上一般接收到信号便结束),而不是遇到取消点才能被取消。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值