Win32线程——优先权
《Win32多线程程序设计》–Jim Beveridge & Robert Wiener
Win32 优先权是以数值表现的,并以进程的“优先权类别(priority class)”、线程的“优先权层级 (priority level)”和操作系统当时采用的“动态提升(Dynamic Boost)”作为计算基准。 所有因素放在一起,最后获得一个0~31 的数值。拥有最高优先权之线程,即为下一个将执行起来的线程。如果你有一大把 worker 线程,其“优先权类别”和“优先权层级”都相同,那么就每一个轮流执行。这是所谓的 “round robin” 调度方式。如果你有一个线程总是拥有最高优先权,那么它就永远获得 CPU 时间,别人都别玩了。这就是为什么必须明智而谨慎地使用优先权的原因。
优先权类别( Priority Class)
“优先权类别”是进程的属性之一。这个属性可以表现出这一进程和其他进程比较之下的重要性。
| 优先权类别(Priority Classes) | 基础优先权值(base priority) |
|---|---|
| HIGH_PRIORITY_CLASS | 13 |
| IDLE_PRIORITY_CLASS | 4 |
| NORMAL_PRIORITY_CLASS | 7 or 8(译注:有些资料上写 7 or 9) |
| REALTIME_PRIORITY_CLASS | 24 |
大部分程序:使用 NORMAL_PRIORITY_CLASS 。少数情况下才会考虑使用其他类别。
Task Manager:使用 HIGH_ PRIORITY_CLASS,所以即使其他程序处于非常忙碌的状态下,它也总是能够有所反应。
屏幕保护程序(screen saver ):使用IDLE_PRIORITY_CLASS,只会在 CPU 绝对空闲的时候才执行。
和时间有密切关系的程序:使用REALTIME_ PRIORITY_CLASS。可以使该进程甚至优于核心进程和设备驱动程序。这个优先权类别不应该用于标准 GUI 程序或甚至于典型的服务器程序。
优先权层级(Priority Level)
调整同一个进程内的各线程的相对重要性。
| 优先权层级(Priority Levels) | 调整值 |
|---|---|
| THREAD_PRIORITY_HIGHEST | +2 |
| THREAD_PRIORITY_ABOVE_NORMAL | +1 |
| THREAD_PRIORITY_NORMAL | 0 |
| THREAD_PRIORITY_BELOW_NORMAL | –1 |
| THREAD_PRIORITY_LOWEST | –2 |
| THREAD_PRIORITY_IDLE | Set to 1 |
| THREAD_PRIORITY_TIME_CRITICAL | Set to 15 |
注意: 对于 REALTIME_PRIORITY_CLASS 的调整值,有点不同于上表所列。
BOOL SetThreadPriority(HANDLE hThread, int nPriority); int GetThreadPriority(HANDLE hThread);
动态提升( Dynamic Boost)
决定线程真正优先权的最后一个因素是其目前的动态提升值(Dynamic Boost)。所谓动态提升是对优先权的一种调整,使系统能够机动对待线程,以强化程序的可用性。
Windows系统中的“动态提升”被设定为最大:这使得拥有键盘焦点的程序(前台程序)的优先权得以提升 +2 。这个设定使得前台程序比后台程序获得较多的 CPU 时间,因此即使系统忙碌,前台程序还是容易保持其 UI 敏感度。
一个进程的线程:只要线程获得键盘输入,该线程就得到一个 +5 的优先权调整值。这使得该线程有机会处理那个输入,并且提供立即的回应给用户。其他可能引起优先权动态提升的情况还包括鼠标消息、计时器消息等等。
任何一个线程(不限属于哪一个进程):那是在一个“等待状态”获得满足时发生的,例如有一个线程正在等待一个 mutex,当 Wait…() 返回时,该线程的优先权会获得动态提升。这样的提升意味着 critical sections 将尽可能地被快速处理,而等待时间将尽可能地缩短。
(示例:创建初始挂起线程,设置优先级后取消挂起执行)
#include <stdio.h>
#include <time.h>
#include <Windows.h> DWORD WINAPI Thread(void *arg) {
for (int i = ; i < ; i++) {
printf("Run #%d\n", (int)time(NULL));
Sleep();
}
return ;
} int main(void) {
HANDLE hThread = CreateThread(NULL, , Thread, NULL, CREATE_SUSPENDED, NULL); // 创建线程,挂起不执行
SetThreadPriority(hThread, THREAD_PRIORITY_IDLE); // 设置线程优先级 Sleep();
printf("resume hThread\n");
ResumeThread(hThread); // 3s后继续执行线程hThread Sleep();
printf("suspend hThread\n");
SuspendThread(hThread); // 3s后挂起不执行线程hThread Sleep();
printf("resume hThread\n");
ResumeThread(hThread); // 3s后继续执行线程hThread WaitForSingleObject(hThread, INFINITE);
CloseHandle(hThread);
return ;
}

参考链接:Win32线程——优先权
Win32线程——优先权的更多相关文章
- 第5章 不要让线程成为脱缰的野马(Keeping your Threads on Leash) ---线程优先权(Thread priority)
有没有过这样的经验?你坐在你的车子里,目的地还在好几公里之遥,而时间已经很晚了.你拼命想告诉那些挡住你去路的人们,今天这个约会对你是多么多么重要,能不能请他们统统--呃--滚到马路外?很不幸,道路系统 ...
- win32线程池代码(WinApi/C++)
win32线程池代码(WinApi/C++) 健壮, 高效,易用,易于扩, 可用于任何C++编译器 //说明, 这段代码我用了很久, 我删除了自动调整规模的代码(因为他还不成熟)/********** ...
- win32线程
win32线程 一丶什么是线程 在windows中常听到的就是线程.多线程.啊什么的. 这里介绍一下什么是线程. 1.线程是附属在进程中的一个执行实体.简而言之就是执行代码的. 2.每个进程至少有一个 ...
- Win32 线程同步
Win32 线程同步 ## Win32线程同步 ### 1. 原子锁 ### 2. 临界区 {全局变量} CRITICAL_SECTION CS = {0}; // 定义并初始化临界区结构体变量 {线 ...
- win32线程栈溢出问题 (一)
一.什么是线程栈溢出 我们都知道,每一个win32线程都会开辟一个空间,用来临时存储线程执行时所调用的一系列函数的参数.返回地址和局部变量及其他上下文信息.这个空间就是线程的栈区.栈区的容量是有限的, ...
- Win32线程——等待另一个线程结束
转载: https://blog.csdn.net/yss28/article/details/53646627 <Win32多线程程序设计>–Jim Beveridge & Ro ...
- C++ win32线程数上限
hThread = CreateThread( NULL, 0, WorkerFunction, &threadParm, 0, &dwThreadID ); 这样的创建方法 ...
- win32 线程通信初步
// 线程通信机制.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #define NUM_THREADS 10 #include < ...
- Win32线程安全问题.同步函数
线程安全问题.同步函数 一丶简介什么是线程安全 通过上面几讲.我们知道了线程怎么创建.线程切换的原理(CONTEXT结构) 每个线程在切换的时候都有自己的堆栈. 但是这样会有安全问题. 为什么? 我 ...
随机推荐
- Java - 多线程中的不变性问题
这篇记录一下保证并发安全性的策略之——不变性. (注意:是Immutable,不是Invariant!) 将一连串行为组织为一个原子操作以保证不变性条件,或者使用同步机制保证可见性,以防止读到失效数据 ...
- DataGridView初始化,加载数据
1,创建winform窗体应用程序 2,在界面上拖入DataGridView控件 3,添加相应的列如图: 4,开始编写后面的代码: private DataTable CountryDt = new ...
- 键盘录入(Java)
键盘录入(Java): 1.导包 格式 import java.util.Scanner; 位置 在class上面 2.创建键盘录入对象 格式 Scanner sc = new Scanner(Sys ...
- 模块与包&常用模块
一.模块的使用 模块定义:一系列功能的集合体 分为三大类:1.自定义模块 2.内置模块(比如 time,os,sys) 3.第三方模块 模块的表现形式: 1.使用python编写的py文件 2.已被编 ...
- 软件项目技术点(6)——结合鼠标操作绘制动态canvas画布
AxeSlide软件项目梳理 canvas绘图系列知识点整理 我们创建一个类封装了所有鼠标需要处理的事件. export class MouseEventInfo { el: HTMLElemen ...
- CSS性能优化新属性:will-change
---恢复内容开始--- will-change属性通过告诉浏览器什么属性.什么元素将会发生变化,可以对这些操作进行可能性的优化,由此提高CSS动画的执行效率. 这个属性可以有4个值: auto: 实 ...
- 实现移动端touch事件的横向滑动列表效果
要实现手机端横向滑动效果并不难,了解实现的原理及业务逻辑就很容易实现.原理:touchstart(手指按下瞬间获取相对于页面的位置)——>touchmove(手指移动多少,元素相应移动多少). ...
- Python基础-继承与派生
一.继承 继承是一种创建新的类的方式,在python中,新建的类可以继承自一个或者多个父类,原始类称为基类或超类,新建的类称为派生类或子类. python中类的继承分为:单继承和多继承 class P ...
- CentOS新增硬盘,重新扫描总线
Centos 新增硬盘以后,系统不能自动进行识别. 1. 由于不知道新增硬盘挂载的位置,可以先查看现有硬盘挂载的适配器. [root@localhost ~]# ls -l /sys/block/sd ...
- <Android 应用 之路> 百度地图API使用(1)
简介 详情请看百度地图官方网站 http://lbsyun.baidu.com/index.php?title=androidsdk/guide/introduction 使用方式 申请密钥,针对移动 ...