#include
#include
#include
#pragma comment( lib, "Winmm" )
static int counter = 0;
static int64_t ticks_per_second;
void __stdcall on_timer(HWND h, UINT ui, UINT_PTR up, DWORD dw)
{
std::cout = 1.0)
Sleep((DWORD)ms_need_sleep);
else
continue;
}
}
这里主要用到的几个 win32api 为
-
MMRESULT timeBeginPeriod
使用该 api 需要链接Winmm
, 所以我们在文件顶部加入#pragma comment( lib, "Winmm" )
.
它的作用是请求提高一些计时器的精度比如这里的Sleep
, 默认 windows 似乎只会给我们提供 10ms 左右很粗糙的精度, 所以这里我们直接请求尽可能的高的精度, 即1ms
. -
QueryPerformanceFrequency
该 api 用于获取”性能计时器”的精度, 单位是 ticks每秒. 在我的机子上它的值是10000000
, 可以看到精度还是很令人满意的. 在这里我们将其与expected
(也就是期望每次调用的间隔,单位s)进行相乘, 得到一个以ticks为单位的间隔. -
QueryPerformanceCounter
该 api 会检测”性能计时器”的值, 单位为 ticks, 在 msdn 中其精度的描述为
那么结合上述几个 api 以及几个简单的数学运算, 这样就可以相对稳定的定时调用函数了(在这里是 1s 60 次):
time out, counter=59
time out, counter=59
time out, counter=60
time out, counter=59
time out, counter=58
time out, counter=60
当你注释掉timeBeginPeriod
的调用后你会发现结果不是很乐观(即使我们期望 1s 调用 60 次):
time out, counter=33
time out, counter=31
time out, counter=32
time out, counter=31
最后, 这个可能常见于游戏的帧率控制, 实际上我就是从这里知道的这些东西(x
服务器租用托管,机房租用托管,主机租用托管,https://www.e1idc.com