zoukankan      html  css  js  c++  java
  • Linux c编程:线程属性

    前面介绍了pthread_create函数,并且当时的例子中,传入的参数都是空指针,而不是指向pthread_attr_t结构的指针。可以使用pthread_attr_t结构修改线程默认属性,并把这些属性与创建的线程联系起来。可以使用pthread_attr_init函数初始化pthread_attr_t结构。调用pthread_attr_init以后,pthread_attr_t结构所包含的内容就是操作系统实现支持的线程所有属性的默认值。如果要修改其中个别属性的值,需要调用其他的函数。

    #include <pthread.h>

     

    int pthread_attr_init( pthread_attr_t *attr );

     

    int pthread_attr_destroy( pthtread_attr_t *attr );

     

    两个函数的返回值都是:若成功则返回0,否则返回错误编号

     

     

    如果要去除对pthread_attr_t结构的初始化,可以调用pthread_attr_destroy函数。如果pthread_attr_init实现时为属性对象分配了动态内存空间,pthread_attr_destroy将会释放该内存空间。除此之外,pthread_attr_destroy还会用无效的值初始化属性对象,因此如果该属性对象被误用,将会导致pthread_create函数返回错误。

    pthread_attr_t结构对应用程序是透明的,也就是说应用程序并不需要了解有关属性对象内部结构的任何细节,因而可以增强应用程序的可移植性。POSIX.1沿用了这种模型,并且为查询和设置每种属性定义了独立的函数

     名称

      描述

      detachstate

      线程的分离状态属性

      guardsize

      线程栈末尾的警戒缓冲区大小(字节数)

      stackaddr

      线程栈的最低地址

      stacksize

      线程栈的大小(字节数)

    如果对现有的某个线程的终止状态不感兴趣的话,可以使用pthread_detach函数让操作系统在线程退出时收回它所占用的资源。

    如果在创建线程时就知道不需要了解线程的终止状态,则可以修改pthread_attr_t结构中的detachstate线程属性,让线程以分离状态启动。可以使用pthread_attr_setdetachstate函数把线程属性detachstate设置为下面的两个合法值之一:设置为PTHREAD_CREATE_DETACHED,以分离状态启动线程;或者设置为PTHREAD_CREATE_JOINABLE,正常启动线程,应用程序可以获取线程的终止状态。

    #include <pthread.h>
     
    int pthread_attr_getdetachstate( const pthread_attr_t *restrict attr,
                                     int *detachstate );
     
    int pthread_attr_setdetachstate( pthread_attr_t *attr, int detachstate );
     
    两者的返回值都是:若成功则返回0,否则返回错误编号
     
    可以调用pthread_attr_getdetachstate函数获取当前的detachstate线程属性,第二个参数所指向的整数用来保存获取到的detachstate属性值:PTHREAD_CREATE_DETACHED,或PTHREAD_CREATE_JOINABLE)。
     
    实例
     
    程序:以分离状态创建的线程
     
    复制代码
    #include "apue.h"
    #include <pthread.h>
     
    int
    makethread(void *(*fn)(void *), void *arg)
    {
        int            err;
        pthread_t       tid;
        pthread_attr_t    attr;
     
        err = pthread_attr_init(&attr);
        if(err != 0)
            return(err);
        err = pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);
        if(err == 0)
            err = pthread_create(&tid, &attr, fn, arg);
        pthread_attr_destroy(&attr);
        return(err);
    }
     
    这里忽略了pthread_attr_destroy函数调用的返回值。在这种情况下,由于对线程属性进行了合理的初始化,pthread_attr_destroy一般不会失败。但是如果pthread_attr_destroy确实出现了失败的情况,清理工作就会变得很困难:必须销毁刚刚创建的线程,而这个线程可能已经运行,并且与pthread_attr_destroy函数可能是异步执行的。忽略pthread_attr_destroy的错误返回可能出现的最坏的情况是:如果pthread_attr_init分配了内存空间,这些内存空间会被泄露。另一方面,如果pthread_attr_init成功地对线程属性进行了初始化,但pthread_attr_destroy在做清理工作时却出现了失败,就没有任何补救策略,因为线程属性结构对应用程序来说是透明的,可以对线程属性结构进行清理的唯一接口是pthread_attr_destroy,但它失败了。
     
    对于遵循POSIX标准的操作系统来说,并不一定要支持线程栈属性,但是对于遵循XSI的系统,支持线程栈属性就是必须的。可以在编译阶段使用_POSIX_THREAD_ATTR_STACKADDR和_POSIX_ATTR_STACKSIZE符号来检查系统是否支持线程栈属性,如果系统定义了这些符号,就说明它支持相应的线程栈属性。也可以通过在运行阶段把_SC_THREAD_ATTR_STACKADDR和_SC_THREAD_ATTR_STACKSIZE参数传给sysconf函数,检查系统对线程栈属性的支持情况。
     
    POSIX.1定义了线程栈属性的一些操作接口。线程栈属性的查询和修改一般是通过较新的函数pthread_attr_getstack和pthread_attr_setstack来进行。
     
     
    #include <pthread.h>
     
    int pthread_attr_getstack( const pthread_attr_t *restrict attr,
                           void **restrict stackaddr,
                           size_t *restrict stacksize );
     
    int pthread_attr_setstack( const pthread_attr_t *attr,
                               void *stackaddr, size_t *stacksize );
     
    两者的返回值都是:若成功则返回0,否则返回错误编号
     
     
    如果希望改变栈的默认大小,但又不想自己处理线程栈的分配问题,这时使用pthread_attr_setstacksize函数就非常有用。
     
    线程属性guardsize控制着线程栈末尾之后用以避免栈溢出的扩展内存的大小。这个属性默认设置为PAGESIZE个字节。可以把guardsize线程属性设为0,从而不允许属性的这种特征行为发生:在这种情况下不会提供警戒缓冲区。同样地,如果对线程属性stackaddr作了修改,系统就会假设我们会自己管理栈,并使警戒栈缓冲区机制无效,等同于把guardsize属性设为0。
     
    复制代码
    #include <pthread.h>
     
    int pthread_attr_getguardsize( const pthread_attr_t *restrict attr, 
                            size_t *restrict guardsize );
     
    int pthread_attr_setguardsize( pthread_attr_t *attr, size_t guardsize );
     
    两者的返回值都是:若成功则返回0,否则返回错误编号
     
     
    如果guardsize线程属性被修改了,操作系统可能把它取为页大小的整数倍。如果线程的栈指针溢出到警戒区域,应用程序就可能通过信号接收到出错信息。
  • 相关阅读:
    eclipse workspace
    logcat and monkey
    git command
    Take Total Control of Internet Explorer with Advanced Hosting Interfaces
    #import 指令
    今天聊发兴致,写了一个 COM STEP BY STEP,结果。。。
    DECLARE_CLASSFACTORY_SINGLETON 宏
    对象微操
    宏定义缺失的解决
    读取地址本内容
  • 原文地址:https://www.cnblogs.com/zhanghongfeng/p/9387549.html
Copyright © 2011-2022 走看看