如何在linux/unix中设置线程的优先级
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_OTHER
Show current configuration of priority
max_priority = 0
min_priority = 0
Show SCHED_FIFO of priority
max_priority = 99
min_priority = 1
Show SCHED_RR of priority
max_priority = 99
min_priority = 1
Show priority of current thread
priority = 0
Set thread policy
Set SCHED_FIFO policy
policy = SCHED_FIFO
Set SCHED_RR policy
policy = SCHED_RR
Restore current policy
policy = SCHED_OTHER
我们可以看到
SCHED_OTHER 是不支持优先级使用的,而SCHED_FIFO和SCHED_RR支持优先级的使用,他们分别为1和99,数值越大优先级越高。从上面的结果我们可以看 出,如果程序控制线程的优先级,一般是用pthread_attr_getschedpolicy来获取系统使用的调度策略,如果是 SCHED_OTHER的话,表明当前策略不支持线程优先级的使用,否则可以。当然所设定的优先级范围必须在最大和最小值之间。我们可以通过 sched_get_priority_max和sched_get_priority_min来获取。
可能网友会问,是否我们可以通过 int pthread_attr_setschedpolicy(pthread_attr_t *attr, int policy);来设定自己所需的调度策略呢?我觉得是完全可以的(有些系统需要定义 _POSIX_THREAD_PRIORITY_SCHEDULING),只要系统实现了对应的调用策略。
说了半天,我们还没有说,在系统允许使用线程优先级别的时候,如何设置优先级别呢?
int pthread_attr_setschedparam(pthread_attr_t *attr, const struct sched_param *param);
int pthread_attr_getschedparam(const pthread_attr_t *attr, struct sched_param *param);
上面两个函数分别用于设置线程的优先级,struct sched_param的定义如下
struct sched_param
{
int __sched_priority; //所要设定的线程优先级
};
例:创建优先级为10的线程
pthread_attr_t attr;
struct sched_param param;
pthread_attr_init(&attr);
pthread_attr_setschedpolicy(&attr, SCHED_RR);
param.sched_priority = 10;
pthread_attr_setschedparam(&attr, ¶m);
pthread_create(xxx , &attr , xxx , xxx);
pthread_attr_destroy(&attr);
linux并不是实时操作系统,把下面的代码运行一遍就能够理解了,代码有很详细的注释。
- #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, ¶m);
- 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);
- long long i;
- while (1) {
- i++;
- i *= 2;
- }
- pthread_exit(NULL);
- }
- int main(void)
- {
- 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/n%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/n%s/n", strerror(ret));
- exit(1);
- }
- policy = SCHED_FIFO;//在Ubuntu9.10上需要root权限
- //设置线程调度策略
- ret = pthread_attr_setschedpolicy(&attr, policy);
- if (ret!=0)
- {
- printf(" pthread_attr_setschedpolicy/n%s/n", strerror(ret));
- exit(1);
- }
- param.sched_priority = 3;
- //设置调度参数
- ret = pthread_attr_setschedparam(&attr, ¶m);
- if (ret!=0)
- {
- printf(" pthread_attr_setschedparam/n%s/n", strerror(ret));
- exit(1);
- }
- //创建线程
- ret = pthread_create(&tid, &attr, thr_fun, NULL);
- if (ret!=0)
- {
- printf("pthread_create/n%s/n", strerror(ret));
- exit(1);
- }
- while (1) {
- printf("hello world/n");
- }
- pthread_join(tid, NULL);
- pthread_exit(NULL);
- }
线程中不要用perror,请使用strerror。