方式七:对于精确度要求更高的定时操作,则应该使用 QueryPerformanceFrequency()
和
QueryPerformanceCounter()函数。这两个函数是 VC 提供的仅供 Windows
95 及其后续版本使用的精确时间函数,并要求计算机从硬件上支持精确定时器。如示例工程中的
Timer7、Timer7_1、Timer7_2、Timer7_3。
QueryPerformanceFrequency()函数和 QueryPerformanceCounter()函数的原型如下:
BOOL QueryPerformanceFrequency(LARGE_INTEGER *lpFrequency);
BOOL QueryPerformanceCounter(LARGE_INTEGER *lpCount); 数据类型
ARGE_INTEGER 既可以是一个 8 字节长的整型数,也可以是两个 4
字节长的整型数的联合结构,
其具体用法根据编译器是否支持 64 位而定。该类型的定义如下:
typedef union _LARGE_INTEGER
{
struct
{
DWORD LowPart ;// 4 字节整型数
LONG HighPart;// 4 字节整型数
};
LONGLONG QuadPart ;// 8 字节整型数
}LARGE_INTEGER ; 在进行定时之前,先调用 QueryPerformanceFrequency()函数获得机
器内部定时器的时钟频率,
然后在需要严格定时的事件发生之前和发生之后分别调用 QueryPerformanceCounter()函数,
利用两次获得的计数之差及时钟频率,计算出事件经
历的精确时间。下列代码实现 1ms 的精确定时:
LARGE_INTEGER litmp;
LONGLONG QPart1,QPart2;
double dfMinus, dfFreq, dfTim;
QueryPerformanceFrequency(&litmp);
dfFreq = (double)litmp.QuadPart;// 获得计数器的时钟频率
QueryPerformanceCounter(&litmp);
QPart1 = litmp.QuadPart;// 获得初始值
do
{
QueryPerformanceCounter(&litmp);
QPart2 = litmp.QuadPart;//获得中止值
dfMinus = (double)(QPart2-QPart1);
dfTim = dfMinus / dfFreq;// 获得对应的时间值,单位为秒
}while(dfTim<0.001); 其定时误差不超过 1 微秒,精度与 CPU
等机器配置有关。
下面的程序用来测试函数 Sleep(100)的精确持续时间:
LARGE_INTEGER litmp;
LONGLONG QPart1,QPart2;
double dfMinus, dfFreq, dfTim;
QueryPerformanceFrequency(&litmp);
dfFreq = (double)litmp.QuadPart;// 获得计数器的时钟频率
QueryPerformanceCounter(&litmp);
QPart1 = litmp.QuadPart;// 获得初始值
Sleep(100);
QueryPerformanceCounter(&litmp);
QPart2 = litmp.QuadPart;//获得中止值
dfMinus = (double)(QPart2-QPart1);
dfTim = dfMinus / dfFreq;//
获得对应的时间值,单位为秒
由于
Sleep()函数自身的
误差,上述程序每次执行的结果都会有微小误差。下列代码实现
1 微秒的精确定时:
LARGE_INTEGER litmp;
LONGLONG QPart1,QPart2;
double dfMinus, dfFreq, dfTim;
QueryPerformanceFrequency(&litmp);
dfFreq = (double)litmp.QuadPart;// 获得计数器的时钟频率
QueryPerformanceCounter(&litmp);