`

linux c学习笔记----线程属性

阅读更多

 

一.线程属性

       线程具有属性,用pthread_attr_t表示,在对该结构进行处理之前必须进行初始化,在使用后需要对其去除初始化。我们用pthread_attr_init函数对其初始化,用pthread_attr_destroy对其去除初始化。

 

名称:

pthread_attr_init/pthread_attr_destroy

功能:

对线程属性初始化/去除初始化

头文件:

#include <pthread.h>

函数原形:

int pthread_attr_init(pthread_attr_t *attr);

int pthread_attr_destroy(pthread_attr_t *attr);

参数:

Attr   线程属性变量

返回值:

若成功返回0,若失败返回-1

      

调用pthread_attr_init之后,pthread_t结构所包含的内容就是操作系统实现支持的线程所有属性的默认值。

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

 

线程属性结构如下:

typedef struct

{

       int                               detachstate;   线程的分离状态

       int                               schedpolicy;  线程调度策略

       struct sched_param              schedparam;  线程的调度参数

       int                               inheritsched;  线程的继承性

       int                                scope;       线程的作用域

       size_t                           guardsize;   线程栈末尾的警戒缓冲区大小

       int                                stackaddr_set;

       void *                          stackaddr;   线程栈的位置

       size_t                           stacksize;    线程栈的大小

}pthread_attr_t;

 

每个个属性都对应一些函数对其查看或修改。下面我们分别介绍。

 


二、线程的分离状态

       线程的分离状态决定一个线程以什么样的方式来终止自己。在默认情况下线程是非分离状态的,这种情况下,原有的线程等待创建的线程结束。只有当pthread_join()函数返回时,创建的线程才算终止,才能释放自己占用的系统资源。

而分离线程不是这样子的,它没有被其他的线程所等待,自己运行结束了,线程也就终止了,马上释放系统资源。程序员应该根据自己的需要,选择适当的分离状态。所以如果我们在创建线程时就知道不需要了解线程的终止状态,则可以pthread_attr_t结构中的detachstate线程属性,让线程以分离状态启动。

 


名称:

pthread_attr_getdetachstate/pthread_attr_setdetachstate

功能:

获取/修改线程的分离状态属性

头文件:

#include <pthread.h>

函数原形:

int pthread_attr_getdetachstate(const pthread_attr_t * attr,int *detachstate);

int pthread_attr_setdetachstate(pthread_attr_t *attr,int detachstate);

参数:

Attr   线程属性变量

Detachstate  线程的分离状态属性

返回值:

若成功返回0,若失败返回-1

 可以使用pthread_attr_setdetachstate函数把线程属性detachstate设置为下面的两个合法值之一:设置为PTHREAD_CREATE_DETACHED,以分离状态启动线程;或者设置为PTHREAD_CREATE_JOINABLE,正常启动线程。可以使用pthread_attr_getdetachstate函数获取当前的datachstate线程属性。

 

以分离状态创建线程

#iinclude <pthread.h>

 

void *child_thread(void *arg)

{

printf(“child thread run!\n”);

}

 

int main(int argc,char *argv[ ])

{

      pthread_t tid;

      pthread_attr_t attr;

 

      pthread_attr_init(&attr);

      pthread_attr_setdetachstate(&attr,PTHREAD_CREATE_DETACHED);

      pthread_create(&tid,&attr,fn,arg);

      pthread_attr_destroy(&attr);

      sleep(1);

}

 

 

三、线程的继承性

       函数pthread_attr_setinheritschedpthread_attr_getinheritsched分别用来设置和得到线程的继承性,这两个函数的定义如下:

 

名称:

pthread_attr_getinheritsched

pthread_attr_setinheritsched

功能:

获得/设置线程的继承性

头文件:

#include <pthread.h>

函数原形:

int pthread_attr_getinheritsched(const pthread_attr_t *attr,int *inheritsched);

int pthread_attr_setinheritsched(pthread_attr_t *attr,int inheritsched);

参数:

attr            线程属性变量

inheritsched     线程的继承性

返回值:

若成功返回0,若失败返回-1

      

 

 这两个函数具有两个参数,第1个是指向属性对象的指针,第2个是继承性或指向继承性的指针。继承性决定调度的参数是从创建的进程中继承还是使用在schedpolicyschedparam属性中显式设置的调度信息。Pthreads不为inheritsched指定默认值,因此如果你关心线程的调度策略和参数,必须先设置该属性。

       继承性的可能值是PTHREAD_INHERIT_SCHED(表示新现成将继承创建线程的调度策略和参数)和PTHREAD_EXPLICIT_SCHED(表示使用在schedpolicyschedparam属性中显式设置的调度策略和参数)。

       如果你需要显式的设置一个线程的调度策略或参数,那么你必须在设置之前将inheritsched属性设置为PTHREAD_EXPLICIT_SCHED.

       下面我来讲进程的调度策略和调度参数。我会结合下面的函数给出本函数的程序例子。


 

四、线程的调度策略

       函数pthread_attr_setschedpolicypthread_attr_getschedpolicy分别用来设置和得到线程的调度策略。

 

名称:

pthread_attr_getschedpolicy

pthread_attr_setschedpolicy

功能:

获得/设置线程的调度策略

头文件:

#include <pthread.h>

函数原形:

int pthread_attr_getschedpolicy(const pthread_attr_t *attr,int *policy);

int pthread_attr_setschedpolicy(pthread_attr_t *attr,int policy);

参数:

attr           线程属性变量

policy         调度策略

返回值:

若成功返回0,若失败返回-1

      

 

 

 这两个函数具有两个参数,第1个参数是指向属性对象的指针,第2个参数是调度策略或指向调度策略的指针。调度策略可能的值是先进先出(SCHED_FIFO)、轮转法(SCHED_RR,或其它(SCHED_OTHER)。

       SCHED_FIFO策略允许一个线程运行直到有更高优先级的线程准备好,或者直到它自愿阻塞自己。在SCHED_FIFO调度策略下,当有一个线程准备好时,除非有平等或更高优先级的线程已经在运行,否则它会很快开始执行。

    SCHED_RR(轮循)策略是基本相同的,不同之处在于:如果有一个SCHED_RR

策略的线程执行了超过一个固定的时期(时间片间隔)没有阻塞,而另外的SCHED_RRSCHBD_FIPO策略的相同优先级的线程准备好时,运行的线程将被抢占以便准备好的线程可以执行。

    当有SCHED_FIFOSCHED_RR策赂的线程在一个条件变量上等持或等持加锁同一个互斥量时,它们将以优先级顺序被唤醒。即,如果一个低优先级的SCHED_FIFO线程和一个高优先织的SCHED_FIFO线程都在等待锁相同的互斥且,则当互斥量被解锁时,高优先级线程将总是被首先解除阻塞。

 

 

 

五、线程的调度参数

       函数pthread_attr_getschedparam pthread_attr_setschedparam分别用来设置和得到线程的调度参数。

 

名称:

pthread_attr_getschedparam

pthread_attr_setschedparam

功能:

获得/设置线程的调度参数

头文件:

#include <pthread.h>

函数原形:

int pthread_attr_getschedparam(const pthread_attr_t *attr,struct sched_param *param);

int pthread_attr_setschedparam(pthread_attr_t *attr,const struct sched_param *param);

参数:

attr           线程属性变量

param          sched_param结构

返回值:

若成功返回0,若失败返回-1

      

 

 

 

 

 

 这两个函数具有两个参数,第1个参数是指向属性对象的指针,第2个参数是sched_param结构或指向该结构的指针。结构sched_param在文件/usr/include /bits/sched.h中定义如下:

      

struct sched_param

{

       int sched_priority;

};

 

结构sched_param的子成员sched_priority控制一个优先权值,大的优先权值对应高的优先权。系统支持的最大和最小优先权值可以用sched_get_priority_max函数和sched_get_priority_min函数分别得到。

 

注意:如果不是编写实时程序,不建议修改线程的优先级。因为,调度策略是一件非常复杂的事情,如果不正确使用会导致程序错误,从而导致死锁等问题。如:在多线程应用程序中为线程设置不同的优先级别,有可能因为共享资源而导致优先级倒置。

 

名称:

sched_get_priority_max

sched_get_priority_min

功能:

获得系统支持的线程优先权的最大和最小值

头文件:

#include <pthread.h>

函数原形:

int sched_get_priority_max(int policy);

int sched_get_priority_min(int policy);

参数:

policy           系统支持的线程优先权的最大和最小值

返回值:

若成功返回0,若失败返回-1

      

 

 

 

 

 下面是上面几个函数的程序例子:

 

#include <pthread.h>
#include <sched.h>
#include <stdio.h>
#include <stdlib.h>
 
void *child_thread(void *arg)
{
int policy;
int max_priority,min_priority;
struct sched_param param;
pthread_attr_t attr;
 
pthread_attr_init(&attr);
pthread_attr_setinheritsched(&attr,PTHREAD_EXPLICIT_SCHED);
pthread_attr_getinheritsched(&attr,&policy);
if(policy==PTHREAD_EXPLICIT_SCHED)
    printf("Inheritsched:PTHREAD_EXPLICIT_SCHED\n");
if(policy==PTHREAD_INHERIT_SCHED)
    printf("Inheritsched:PTHREAD_INHERIT_SCHED\n");
 
pthread_attr_setschedpolicy(&attr,SCHED_RR);
pthread_attr_getschedpolicy(&attr,&policy);
if(policy==SCHED_FIFO)
    printf("Schedpolicy:SCHED_FIFO\n");
if(policy==SCHED_RR)
    printf("Schedpolicy:SCHED_RR\n");
if(policy==SCHED_OTHER)
    printf("Schedpolicy:SCHED_OTHER\n");
 
 
sched_get_priority_max(max_priority);
sched_get_priority_min(min_priority);
printf("Max priority:%u\n",max_priority);
printf("Min priority:%u\n",min_priority);
 
param.sched_priority=max_priority;
pthread_attr_setschedparam(&attr,&param);
printf("sched_priority:%u\n",param.sched_priority);
pthread_attr_destroy(&attr);
}
 
int main(int argc,char *argv[ ])
{
pthread_t child_thread_id;
 
pthread_create(&child_thread_id,NULL,child_thread,NULL);
pthread_join(child_thread_id,NULL);
}

 

 

 

 六栈大小(stack size)

当系统中有很多线程时,可能需要减小每个线程栈的默认大小,防止进程的地址空间不够用;当线程调用的函数会分配很大的局部变量或者函数调用层次很深时,可能需要增大线程栈的默认大小。函数pthread_attr_getstacksize和 pthread_attr_setstacksize被提供。

  int pthread_attr_getstacksize ( const pthread_attr_t *attr, size_t *size );

  int pthread_attr_setstacksize ( pthread_attr_t *attr, size_t size );

  函数pthread_attr_getstack和pthread_attr_setstack函数可以同时操作栈地址和栈大小两个属性:

  int pthread_attr_getstack ( const pthread_attr_t *attr, void **stackaddr, size_t *size );

  int pthread_attr_setstack ( pthread_attr_t *attr, void *stackaddr, size_t size );

 

 

七、栈保护区大小(stack guard size)

在线程栈顶留出一段空间,防止栈溢出。当栈指针进入这段保护区时,系统会发出错误,通常是发送信号给线程。该属性默认值是PAGESIZE大小,该属性被设置时,系统会自动将该属性大小补齐为页大小的整数倍。当改变栈地址属性时,栈保护区大小通常清零。

  int pthread_attr_getguardsize ( const pthread_attr_t *attr, size_t *guardsize );

  int pthread_attr_setguardsize ( pthread_attr_t *attr, size_t guardsize );

 

 

 

 

八、争用范围(scope)

建立线程的争用范围(PTHREAD_SCOPE_SYSTEM 或 PTHREAD_SCOPE_PROCESS)。使用 PTHREAD_SCOPE_SYSTEM 时,此线程将与系统中的所有线程进行竞争。使用 PTHREAD_SCOPE_PROCESS 时,此线程将与进程中的其他线程进行竞争。这个又叫绑定状态,PTHREAD_SCOPE_SYSTEM(绑定的)和PTHREAD_SCOPE_PROCESS(非绑定的)。具有不同范围状态的线程可以在同一个系统甚至同一个进程中共存。进程范围只允许这种线程与同一进程中的其他线程争用资源,而系统范围则允许此类线程与系统内的其他所有线程争用资源。对于Solaris系统,实际上,从 Solaris 9 发行版开始,系统就不再区分这两个范围。

  int pthread_attr_getscope(const pthread_attr_t *restrict attr, int *restrict contentionscope);

  int pthread_attr_setscope(pthread_attr_t *attr, int contentionscope);

 

 

 

(9)线程并行级别(concurrency)

应用程序使用 pthread_setconcurrency() 通知系统其所需的并发级别。

  int pthread_getconcurrency(void);

  int pthread_setconcurrency(int new_level);

 

例:创建优先级为50的线程。

  pthread_attr_t attr;

  struct sched_param param;

  pthread_attr_init(&attr);

  pthread_attr_setschedpolicy(&attr, SCHED_RR);

  param.sched_priority = 50;

  pthread_attr_setschedparam(&attr, ¶m);

  pthread_create(&threadid, &attr, &threadfunc, NULL);

  pthread_attr_destroy(&attr);

分享到:
评论

相关推荐

    Linux多线程编程_linux多线程_Linux多线程;应用笔记_columnc9g_

    2. **线程属性**:通过`pthread_attr_init()`和`pthread_attr_set*()`函数可以设置线程属性,如调度策略、栈大小等。 3. **线程函数**:新创建的线程将从指定的函数开始执行,这个函数通常接受一个void类型的指针...

    linux系统编程笔记

    - 包括__clone函数调用、pthreadAPI、线程属性、pthreadcleanup宏、互斥mutex、条件变量等。 7. 内存管理相关函数 - 包含经典的C动态内存管理函数和Linux的内存映像管理函数。 8. 进程间通信IPC - 包括管道、...

    Linux系统编程学习笔记

    ### Linux系统编程学习笔记 #### 一、IO **1.1 标准I/O (stdio)** - **fopen/fclose**: `fopen` 用于打开或创建一个文件,并返回一个指向该文件的 `FILE *` 类型的指针。`fclose` 用于关闭由 `FILE *` 指向的文件...

    Linux学习文档_多线程编程

    基于C语言的多线程编程在Linux环境中尤其常见,因为C语言提供了底层的访问权限,可以更直接地控制系统的资源。 在"Linux学习文档_多线程编程"中,我们可以预期包含以下几个关键知识点: 1. **线程创建**:在Linux...

    linux-kernel-boot-thread.zip_kernel thread

    这篇学习笔记深入探讨了这个主题,旨在帮助读者理解Linux内核启动线程的工作原理、创建方法以及其在系统中的作用。 首先,内核启动线程是内核自身启动的第一个线程,它的主要职责是初始化系统资源,如内存管理、...

    C语言基础学习笔记 .pdf

    【C语言基础学习笔记】 C语言是一种强大的编程语言,由丹尼斯·里奇创造,它在计算机科学领域占据着重要地位。C语言以其简洁、高效和可移植性著称,被广泛应用于操作系统、嵌入式系统、游戏开发、服务器端应用、...

    linux 系统编程 尚观 linux内核驱动开发 笔记

    尚观Linux内核驱动开发笔记不仅涵盖了这些基础知识,还可能包含实践案例、常见问题解析以及高级技术探讨,例如异步I/O、内存管理优化、多线程同步等。通过学习和实践这些内容,开发者可以提升在Linux平台上的系统...

    Professional Linux Kernel Architecture, 精通Linux内核架构

    - **附录C:C语言笔记** - **知识点**:提供了一些C语言编程的建议和技巧,尤其是那些与编写内核模块相关的。 - **重要性**:C语言是编写内核模块的主要语言,因此这部分内容对于内核开发者而言非常重要。 - **...

    linux应用程序开发,读书笔记

    ### Linux应用程序开发基础知识点 #### 一、Linux操作系统概述 **Linux** 是一款免费且开源的类Unix操作系统,...通过上述知识点的学习,开发者能够更好地理解和使用Linux系统,为后续的应用程序开发打下坚实的基础。

    学习《高级Linux环境编程》读书笔记(APUE读书笔记)

    11. 线程:本部分介绍了线程的基本概念、线程的创建与终止、线程同步、线程属性对象、线程安全、线程私有数据等,这些都是编写多线程程序时需要掌握的知识。 12. 线程控制:这部分进一步讨论了线程属性对象、线程...

    java学习笔记整理

    ### Java学习笔记整理 #### 1. Java简介 ##### 1.1 计算机编程和开发语言 计算机系统由硬件系统和软件系统组成。软件系统又进一步细分为系统软件和应用软件。其中,系统软件包括操作系统、编译系统、数据库系统等...

    Professional Linux Kernel Architecture

    - **附录C:C语言笔记** - **主要内容**:介绍编写内核模块时需要掌握的C语言高级特性。 - **学习目标**:提高C语言编程技能,学会编写高效的内核代码。 - **附录D:系统启动** - **主要内容**:讲述Linux内核的...

    1-java相关课程系列笔记之一java学习笔记.doc

    5. **Linux命令与相关知识**:虽然Java可在多种操作系统上运行,但学习一些基本的Linux命令对于开发者来说非常有用,例如文件操作、进程管理和版本控制等。 6. **Eclipse/Myeclipse程序结构**:这些是流行的Java...

    nutch学习笔记之第一天初学

    ### Nutch 学习笔记之第一天初学 在IT领域,特别是搜索引擎开发和技术研究方向,Apache Nutch无疑是一个值得关注的开源项目。Nutch是Apache软件基金会的一个子项目,旨在为开发者提供一个高度可扩展且可定制化的...

    Linux C 系统编程

    Linux C系统编程是计算机科学领域中的一个重要主题,它涉及到操作系统层面的编程,主要使用C语言在Linux环境下进行。这一主题涵盖了多个关键知识点,包括文件操作、进程管理、内存管理、信号处理、网络编程以及系统...

    linux architecture

    - **C语言笔记**:给出了针对C语言编程的一些建议和最佳实践,对于理解和修改内核源码非常有帮助。 - **系统启动过程**:详细介绍了从加电到内核初始化完成这一阶段的操作流程。 - **ELF二进制格式**:解释了Linux中...

Global site tag (gtag.js) - Google Analytics