Linux c 线程属性,线程优先级的修改

线程属性的设置,网上找的文章总感觉不够全面,还是结合man手册查看。

线程属性设置,分两个方式,一种是在创建之前,通过pthread_attr_t 结构体传入,另一种,是线程创建完已经在运行时,通过部分函数设置。一般常见的是创建线程时传NULL,使用默认属性,后续运行时根据需要动态修改,也就第二种方式。
---/--------------补充:设置优先级,除了这里所描述的 sched_setscheduler()函数之外,还有一个函数 int setpriority(int which, id_t who, int prio);  这两个函数都是系统调用(linux man手册,man 1 ,shell 脚本命令,man 2, 系统调用,man 3 库函数), 前者可以修改其调度策略,后者只是修改调度优先级,按照这个对比来看, setpriority是针对进程 ,用于设置进程,进程组,用户进程的优先级【在调度层面,线程才是基本单位,同一个进程下的线程,只是具有相同的进程资源的关系。 所以 setpriority是不是把同属于这一个进程的线程都进行了优先级的设定?还搞不太清楚】)
Linux c 线程属性,线程优先级的修改_第1张图片

一:线程创建前设置属性:

int pthread_create(pthread_t *thread, const pthread_attr_t *attr,

void *(*start_routine) (void *), void *arg);

1.0 第一个参数,用来保存创建好后线程uid

2.0 第二个参数,即线程属性,通常传NULL,表示默认属性,这个属性在创建前可以设置,包括调度策略,栈大小,优先级等等

3.0 第三个参数,即线程入口函数

4.0 第四个参数,传给线程的参数

所以在创建线程前,对 第二个参数 pthred_attr_t 结构体进pthread_attr_t 进行赋值

 

1.0 pthread_attr_init(pthread_attr_t *attr);//使用默认值填充初始化

2.0//获取设置栈大小 //这个属性只能在线程创建前设置,后面不能动态修改了

Linux c 线程属性,线程优先级的修改_第2张图片

int pthread_attr_setstacksize(pthread_attr_t *attr, size_t stacksize);

int pthread_attr_getstacksize(pthread_attr_t *attr, size_t *stacksize);

3.0 //设置获取调度策略

SCHED_FIFO, SCHED_RR, and

SCHED_OTHER //policy支持这三种值,

    1.0SCHED_OTHER 分时调度策略,

    2.0SCHED_FIFO实时调度策略,先到先服务

    3.0,SCHED_RR实时调度策略,时间片轮转 

   SCHED_OTHER(分时调度)是不支持优先级使用的,其他两个实时调度可以

int pthread_attr_setschedpolicy(pthread_attr_t *attr, int policy);

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

4.0 //设置调度参数,目前这个参数里面就一个量,优先级 //用户程序设置实时任务这个优先级,值越大,优先级越高(内核里面优先级是用另外的一个值,具体查看这个文章:http://www.360doc.com/content/13/0801/09/7775902_303961368.shtml)

struct sched_param {

int sched_priority; /* Scheduling priority */

};

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

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

5.0//是否分离属性

int pthread_attr_setdetachstate(pthread_attr_t *attr, int detachstate);

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

 

上面这些都是静态,在创建线程之前的设置。

二:一般,都在线程创建完后,动态地设置:

1.0 设置调度策略和优先级

#include

int sched_setscheduler(pid_t pid, int policy, const struct sched_param *param);

int sched_getscheduler(pid_t pid); //获取线程当前优先级

struct sched_param {

int sched_priority;

};

//获取支持的最大最小值

int sched_get_priority_max(int policy);

int sched_get_priority_min(int policy);

2.0 设置分离属性

int pthread_detach(pthread_t thread);

int pthread_join(pthread_t thread, void **retval);

3.0 设置线程名称

调用   prctl(PR_SET_NAME, “my_threadname”);

线程默认名称,貌似是父进程的名称,未去考究,在linux上我们用 ps -p pid -T 来查看进程号为 pid的子线程的情况,如果用上面的方式设置了线程名称,则会列出名称

关于 prctl 

Linux c 线程属性,线程优先级的修改_第3张图片

process control,  #man prctl 上面的是 (2),说明这是一个系统调用函数,根据他的全名顾名思义就是用来控制 process的,(linux内核里面的实现,线程其实就是进程,只是对进行了一些资源的共享,所以经常说线程是轻量级的进程) 

关于ps 命令, -p 表示进程号, -T表示列出子线程。所以上面设置线程名称,更多的时用来调试多线程程序的时候方便调试的,毕竟在程序里面直接用线程pid就可以唯一确定线程的身份了,但是没有名称看的直接,比如没有设置名称的线程:

Linux c 线程属性,线程优先级的修改_第4张图片

看一下man手册里面详细的关于PR_SET_NAME的介绍

Linux c 线程属性,线程优先级的修改_第5张图片

名称长度不能超过 16bytes(包含 结束符号'\0'),也可以通过库函数 pthread_setname_np来调用,线程的这个属性,可以通过

/proc文件系统来查看,即/proc/self/task/[tid]/comm

 

====20191011补充=================

创建之前设置属性和创建好后修改属性,注意  sched_setscheduler 的第一个参数是 pid_t ,非pthread_t

pthread_t 和pid_t, 是两个不同的标识符,分别可以通过 pthread_self()  和getpid()来获取当前运行线程的相应标识符。所以上面所说线程运行起来后再动态修改优先级,这个修改放在线程体里面调用即可,通过getpid()来获取pid_t. (还没有深入了解关于具体的 pthread_t 和pid_t的区别,看到一种说法是 pthraed_t用来区分表示同一个进程之间的线程,而pid_t才是系统中认为的线程唯一标识符,没有找到关于这两个是否可以相互转换的说法,估计涉及到具体的操作系统)

demo1:创建之前就设置好优先级

//创建线程之前就定好调度优先级 
//需要相关函数头文件 #include #include 
pthread_t pliveThread;

  pthread_attr_t  pthread_pro;
  pthread_attr_init(&pthread_pro);

  pthread_attr_setschedpolicy(&pthread_pro, SCHED_FIFO);
  struct sched_param s_parm;
  s_parm.sched_priority = sched_get_priority_max(SCHED_FIFO);
  pthread_attr_setschedparam(&pthread_pro, &s_parm);

  int ret = 0;
  if(0 != (ret = pthread_create(&pliveThread,&pthread_pro,live555_main,url))){
       printf("craete thread erro: %s",strerror( ret));
        return -1;
  }

demo2:创建好之后,在线程体里面自己提升优先级

    //在待提升优先级的线程体中运行
    int ret =0;
    pid_t pid =getpid();

    int curschdu = sched_getscheduler(pid);
    if(curschdu <0 )
    {
        printf( "getschedu err %s\n",strerror( errno));
    }
    printf("schedu befor %d\n",curschdu);


    struct sched_param s_parm;
    s_parm.sched_priority = sched_get_priority_max(SCHED_FIFO);
    printf("schedu max %d min %d\n",sched_get_priority_max(SCHED_FIFO),sched_get_priority_min(SCHED_FIFO));
    
    ret = sched_setscheduler(pid, SCHED_FIFO, &s_parm);
    if(ret <0)
    {
       printf( "setschedu err %s\n",strerror( errno));
    }

    curschdu = sched_getscheduler(pid);
    

    printf("schedu after %d\n",curschdu);

在android 第三方app的jni 调用中,创建的线程想提高优先级,结果返回 erro: Operation not permitted,没有权限。。。

 

 

你可能感兴趣的:(c库)