posix多线程有感--线程高级编程(线程调度以及优先级设置)

来源:互联网 发布:it outlet 编辑:程序博客网 时间:2024/05/18 03:58
当在属性对象中设置调度策略或优先级时,必须同时设置inheritsched属性(即继承属性设置为:PTHREAD_EXPLICIT_SCHED)。

linux内核的三种调度方法:

  • SCHED_OTHER 分时调度策略
  • SCHED_FIFO实时调度策略,先到先服务
  • SCHED_RR实时调度策略,时间片轮转

实时进程将得到优先调用,实时进程根据实时优先级决定调度权值,分时进程则通过nice和counter值决定权值,nice越小,counter越大,被调度的概率越大,也就是曾经使用了cpu最少的进程将会得到优先调度。

SHCED_RR和SCHED_FIFO的不同:当采用SHCED_RR策略的进程的时间片用完,系统将重新分配时间片,并置于就绪队列尾。放在队列尾保证了所有具有相同优先级的RR任务的调度公平。

SCHED_FIFO一旦占用cpu则一直运行。一直运行直到有更高优先级任务到达或自己放弃。如果有相同优先级的实时进程(根据优先级计算的调度权值是一样的)已经准备好,FIFO时必须等待该进程主动放弃后才可以运行这个优先级相同的任务。而RR可以让每个任务都执行一段时间。

int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void*), void *arg);来创建线程,但是如何设置线程的优先级呢?在讨论这个问题的时候,我们先要确定当前线程使用的调度策略,posix提供了int pthread_attr_getschedpolicy(const pthread_attr_t *attr, int *policy);

函数来获取所使用的调度策略,它们是:SCHED_FIFO, SCHED_RR 和 SCHED_OTHER。

我们可以使用:int sched_get_priority_max(int policy);int sched_get_priority_min(int policy);来获取线程线程可是设置的最大和最小的优先级值,如果调用成功就返回最大和最小的优先级值,否则返回-1。从我现在运行的linux系统中,我使用下列程序(程序见附录)获取了对应三种调度策略中的最大和最小优先级:policy = SCHED_OTHERShow current configuration of prioritymax_priority = 0min_priority = 0Show SCHED_FIFO of prioritymax_priority = 99min_priority = 1Show SCHED_RR of prioritymax_priority = 99min_priority = 1Show priority of current threadpriority = 0Set thread policySet SCHED_FIFO policypolicy = SCHED_FIFOSet SCHED_RR policypolicy = SCHED_RRRestore current policypolicy = SCHED_OTHER

我们可以看到SCHED_OTHER是不支持优先级使用的,而SCHED_FIFO和SCHED_RR支持优先级的使用,他们分别为1和99,数值越大优先级越高。 从上面的结果我们可以看出,如果程序控制线程的优先级,一般是用pthread_attr_getschedpolicy来获取系统使用的调度策略,如果是SCHED_OTHER的话,表明当前策略不支持线程优先级的使用,否则可以。当然所设定的优先级范围必须在最大和最小值之间。我们可以通过sched_get_priority_max和sched_get_priority_min来获取。

#include    <stdio.h>#include    <stdlib.h>#include    <unistd.h>#include    <pthread.h>#include    <signal.h>#include    <string.h>void * thr_fun(void *arg){    int policy, ret;    struct sched_param param;    //获取线程调度参数    ret = pthread_getschedparam(pthread_self(), &policy, &param);    if(ret!=0)    {        printf("pthread_getschedparam %s\n", strerror(ret) );        exit(1);    }    if (policy == SCHED_FIFO)    {        printf("policy:SCHED_FIFO\n");    }    else if (policy == SCHED_OTHER)    {        printf("policy:SCHED_OTHER\n");    }    else if (policy == SCHED_RR)    {        printf("policy:SCHED_RR\n");    }    printf("param: %d\n", param.sched_priority);    printf("pthread getpid = %d\n",getpid());    long long i;    while (1) {        i++;        i *= 2;    }    pthread_exit(NULL);}int main(int argc,char *argv[]){    int ret;    pthread_t tid;    pthread_attr_t attr;    int policy, inher;    struct sched_param param;    //初始化线程属性    pthread_attr_init(&attr);    //获取继承的调度策略    ret = pthread_attr_getinheritsched(&attr, &inher);    if (ret!=0)    {        printf("pthread_attr_getinheritsched %s\n", strerror(ret));        exit(1);    }    if (inher == PTHREAD_EXPLICIT_SCHED)    {        printf("PTHREAD_EXPLICIT_SCHED\n");    }    else if (inher == PTHREAD_INHERIT_SCHED)    {        printf("PTHREAD_INHERIT_SCHED\n");        inher = PTHREAD_EXPLICIT_SCHED;    }    //设置继承的调度策略    //必需设置inher的属性为 PTHREAD_EXPLICIT_SCHED,否则设置线程的优先级会被忽略    ret = pthread_attr_setinheritsched(&attr, inher);    if (ret!=0)    {        printf("pthread_attr_setinheritsched %s\n", strerror(ret));        exit(1);    }    sleep(1);    policy = SCHED_FIFO;//在Ubuntu10.04上需要root权限    //设置线程调度策略    ret = pthread_attr_setschedpolicy(&attr, policy);    if (ret!=0)    {        printf(" pthread_attr_setschedpolicy%s\n", strerror(ret));        exit(1);    }    param.sched_priority = 3;    //设置调度参数    ret = pthread_attr_setschedparam(&attr, &param);    if (ret!=0)    {        printf(" pthread_attr_setschedparam %s\n", strerror(ret));        exit(1);    }    //创建线程    ret = pthread_create(&tid, &attr, thr_fun, NULL);    if (ret!=0)   {        printf("pthread_create %s\n", strerror(ret));        exit(1);    }    printf("main getpid = %d\n",getpid());    sleep(5);    while (1) {        printf("hello world\n");        printf("getpid = %d\n",getpid());    }    pthread_join(tid, NULL);    pthread_exit(NULL);}

root@# top -H
Priority 3 FIFO [99-100-3=-4]???

root@:/proc/3408/task/3409# cat sched  

实时进程优先级:  

prio = MAX_RT_PRIO(100)– 1 – rt_priority(3)= 96;

policy:

0 SCHED_OTHER 
1.SCHED_FIFO 
2.SCHED_RR

原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 腿着凉了特别疼怎么办 孩子骨龄大2两年怎么办 和人吃饭很尴尬怎么办 头不自觉向右偏怎么办 靠墙站立腰疼怎么办 小腿酸痛乏力肌肉萎缩怎么办 搬重物后手臂疼怎么办 和尚鹦鹉吃了盐怎么办 刚买鹦鹉不上手怎么办 word的文件时间改了怎么办 图强gps编码丢失怎么办 武统台湾后岛民怎么办 没有你我怎么办是什么歌 ios 12软件闪退怎么办 来大姨妈想吐怎么办 3岁宝宝体重轻怎么办 硕士延期毕业考上博士怎么办 中国人移民欧洲饮食不习惯怎么办 出车祸了报警警察不管怎么办 高中的孩子不好好上学怎么办 和老公消费观念不合拍怎么办 去医院没带现金怎么办 微信读书下架了怎么办 24岁血压有点高怎么办 吃鸡鼠标弹出来怎么办 电脑分辨率调错了怎么办 猎豹sc9打不开门怎么办 苹果手机卡顿反应慢怎么办 金立手机卡顿反应慢怎么办 20天宝宝黄疸219怎么办? 智慧树选修挂科怎么办 军人被纠察抓了怎么办 我家的小孩很凶怎么办 10岁儿童老挤眼怎么办? 脚趾长鸡眼好痛怎么办 月加班超过36小时怎么办 月经老提前怎么办才好 20岁例假不规律怎么办 农村的医保门诊封顶了怎么办 保研夏令营被拒怎么办 小学生去英国游学怎么办签证