收藏 分享(赏)

Windows计时器.doc

上传人:hwpkd79526 文档编号:8203779 上传时间:2019-06-13 格式:DOC 页数:5 大小:42.50KB
下载 相关 举报
Windows计时器.doc_第1页
第1页 / 共5页
Windows计时器.doc_第2页
第2页 / 共5页
Windows计时器.doc_第3页
第3页 / 共5页
Windows计时器.doc_第4页
第4页 / 共5页
Windows计时器.doc_第5页
第5页 / 共5页
亲,该文档总共5页,全部预览完了,如果喜欢就下载吧!
资源描述

1、Windows 计时器什么时候我们需要用到 SetTimer 函数呢?当你需要每个一段时间执行一件事的的时候就需要使用 SetTimer 函数了。 让我们先来看看 SetTimer 函数的原型: UINT SetTimer(UINT nIDEvent,UINT nElapse,void(CALLBACK EXPORT *lpfnTimer)(HWND,UINT ,YINT ,DWORD) 当使用 SetTimer 函数的时候,就会生成一个计时器。函数中 nIDEvent 指的是计时器的标识,也就是名字。nElapse 指的是时间间隔,也就是每隔多长时间触发一次事件。第三个参数是一个回调函数,在

2、这个函数里,放入你想要做的事情的代码,你可以将它设定为NULL,也就是使用系统默认的回调函数,系统默认认的是 onTime 函数。这个函数怎么生成的呢?你需要在需要计时器的类的生成 onTime 函数:在 ClassWizard 里,选择需要计时器的类,添加 WM_TIME 消息映射,就自动生成 onTime 函数了。然后在函数里添加代码,让代码实现功能。每隔一段时间就会自动执行一次。 例: SetTimer(1,1000,NULL); 1:计时器的名称; 1000:时间间隔,单位是毫秒; NULL:使用 onTime 函数。 当不需要计时器的时候调用 KillTimer(nIDEvent);

3、 例如:KillTimer(1); 2. 或许你会问,如果我要加入两个或者两个以上的 timer 怎么办? 继续用 SetTimer 函数吧,上次的 timer 的 ID 是 1,这次可以是 2,3,4。 SetTimer(2,1000,NULL); SetTimer(3,500,NULL); 嗯,WINDOWS会协调他们的。当然 onTimer 函数体也要发生变化,要在函数体内添加每一个 timer 的处理代码: onTimer(nIDEvent) switch(nIDEvent) case 1:; break; case 2:.; break; case 3:; break; *SDK 编

4、程笔记 计时器篇两个计时器 API 的讨论SetTimer 函数用于创建一个计时器,KillTimer 函数用于销毁一个计时器。计时器属于系统资源,使用完应及时销毁。SetTimer 的函数原型如下:UINT_PTR SetTimer( HWND hWnd, UINT_PTR nIDEvent, UINT uElapse, TIMERPROC lpTimerFunc ) ;其中hWnd 是和 timer 关联的窗口句柄,此窗口必须为调用 SetTimer 的线程所有;如果hWnd 为 NULL,没有窗口和 timer 相关联并且 nIDEvent 参数被忽略nIDEvent 是 timer 的

5、标识,为非零值;如果 hWnd 为 NULL 则被忽略;如果 hWnd 非NULL 而且与 timer 相关联的窗口已经存在一个为此标识的 timer,则此次 SetTimer 调用将用新的 timer 代替原来的 timer。timer 标识和窗口相关,两个不同的窗口可以拥有nIDEvent 相同的 tiemruElapse 是以毫秒指定的计时间隔值,范围为 1 毫秒到 4,294,967,295 毫秒(将近 50天),这个值指示 Windows 每隔多久时间给程序发送 WM_TIMER 消息。lpTimerFunc 是一个回调函数的指针,俗称 TimerFunc;如果 lpTimerFun

6、c 为 NULL,系统将向应用程序队列发送 WM_TIMER 消息;如果 lpTimerFunc 指定了一个值,DefWindowProc 将在处理 WM_TIMER 消息时调用这个 lpTimerFunc 所指向的回调函数,因此即使使用 TimerProc 代替处理 WM_TIMER 也需要向窗口分发消息。关于 SetTimer 的返回值:如果 hWnd 为 NULL,返回值为新建立的 timer 的 ID,如果hWnd 非 NULL,返回一个非 0 整数,如果 SetTimer 调用失败则返回 0KillTimer 的函数原型为:BOOL KillTimer( HWND hWnd, UIN

7、T_PTR uIDEvent ) ; 参数意义同 SetTimer。关于 KillTimer 对消息队列中剩余未处理的 WM_TIMER 消息的影响,MSDN 和Programming Windows 上的说法完全相反。MSDN 的说法很干脆:The KillTimer function does not remove WM_TIMER messages already posted to the message queue. 而petzold 则说 The KillTimer call purges the message queue of any pending WM_TIMER mess

8、ages. Your program will never receive a stray WM_TIMER message following a KillTimer call. (KillTimer 消除消息队列中任何未处理的 WM_TIMER 消息,调用 KillTimer 后你的程序永远不会收到一条“漂泊游荡”的 WM_TIMER 消息)关于 WM_TIMER 消息wParam 为计时器的 ID;如果需要设定多个计时器,那么对每个计时器都使用不同的计时器 ID。wParam 的值将随传递到窗口过程中的 WM_TIMER 消息的不同而不同。lParam 为指向 TimerProc 的指针

9、,如果调用 SetTimer 时没有指定 TimerProc(参数值为 NULL),则 lParam 为 0(即 NULL)。可以通过在窗口过程中提供一个 WM_TIMER case 处理这个消息,或者,默认窗口过程会调用 SetTimer 中指定的 TimerProc 来处理 WM_TIMER 消息使用计时器的三种方法如果在程序的整个执行过程中使用计时器,一般在处理 WM_CREATE 消息时或 WinMain中消息循环前调用 SetTimer,在处理 WM_DESTROY 消息时或在 WinMain 中消息循环后 return前调用 KillTimer。根据 SetTimer 中的参数不同

10、,有三种方法使用计时器。方法一:调用 SetTimer 时指定窗口句柄 hWnd,nIDEvent 中指定计时器 ID,将lpTimerFunc 置 NULL 从而不使用 TimerProc;在窗口过程中处理 WM_TIMER 消息。调用KillTimer 时,使用 SetTimer 中指定的 hWnd 和 id。最好使用#define 定义 timer 的 id,例如:#define ID_TIMER 1SetTimer(hWnd,ID_TIMER,1000,NULL) ;KillTimer(hWnd,ID_TIMER) ;方法二:调用 SetTimer 时指定窗口句柄 hWnd,nIDEv

11、ent 中指定计时器ID,lpTimerFunc 参数不为 NULL 而指定为 TimerProc 函数的指针。这种方法使用TimerProc 函数(名字可自定)处理 WM_TIMER 消息:VOID CALLBACK TimerProc ( HWND hwnd, UINT message, UINT iTimerID, DWORD dwTime)/处理 WM_TIMER 讯息TimerProc 的参数 hwnd 是在调用 SetTimer 时指定的窗口句柄。Windows 只把WM_TIMER 消息送给 TimerProc,因此消息参数总是等于 WM_TIMER。iTimerID 值是计时器

12、ID,dwTimer 值是与从 GetTickCount 函数的返回值相容的值。这是自 Windows 启动后所经过的毫秒数。 使用这种方法时,相关函数调用的形式为:SetTimer(hWnd,ID_TIMER,1000,TimerProc) ;KillTimer(hWnd,ID_TIMER) ;方法三:调用 SetTimer 时不指定窗口句柄(为 NULL),iTimerID 参数自然被忽略,lpTimerFunc 不为 NULL 而指定为 TimerProc 的指针。正如上面 SetTimer 的讨论中所说的,此时 SetTimer 的返回值正是新建立的计时器的 ID,需将这个 ID 保存

13、以供 KillTimer 销毁计时器时所用。当然,KillTimer 的 hWnd 参数也置为 NULL。这种方法同样用 TimerProc 处理 WM_TIMER 消息。UINT_PTR iTimerID ;iTimerID = SetTimer(NULL,0,1000,TimerProc) ;KillTimer(NULL,iTimerID) ;使用这种方法的好处是不必自己指定计时器 ID,这样就不必担心用错 ID。使用多个计时器使用多个计时器只要在建立计时器时指定不同的 ID。比如用上面所述方法一时的情况:#define TIMER_SEC 1#define TIMER_MIN 2然后使用

14、两个 SetTimer 来设定两个计时器:SetTimer (hwnd, TIMER_SEC, 1000, NULL) ;SetTimer (hwnd, TIMER_MIN, 60000, NULL) ;WM_TIMER 的处理如下所示:case WM_TIMER:switch (wParam)case TIMER_SEC:/每秒一次的处理break ;case TIMER_MIN:/每分钟一次的处理break ;return 0 ;改变计时器的时间间隔如果想将一个已经存在的计时器设定为不同的时间间隔,可以简单地用不同的时间值再次调用 SetTimer。计时器精确吗?计时器并不精确。有两个原因

15、:原因一:Windows 计时器是硬件和 ROM BIOS 架构下之计时器一种相对简单的扩充。回到 Windows 以前的 MS-DOS 程序写作环境下,应用程式能够通过拦截者称为 timer tick 的BIOS 中断来实现时钟或计时器。一些为 MS-DOS 编写的程序自己拦截这个硬件中断以实现时钟和计时器。这些中断每 54.915 毫秒产生一次,或者大约每秒 18.2 次。这是原始的IBM PC 的微处理器频率值 4.772720 MHz 被 218 所除而得出的结果。在 Windows 98 中,计时器与其下的 PC 计时器一样具有 55 毫秒的解析度。在 Microsoft Windo

16、ws NT 中,计时器的解析度为 10 毫秒。Windows 应用程式不能以高于这些解析度的频率(在 Windows 98 下,每秒 18.2 次,在 Windows NT 下,每秒大约 100 次)接收 WM_TIMER 消息。在 SetTimer 中指定的时间间隔总是截尾后 tick 数的整数倍。例如,1000 毫秒的间隔除以 54.925 毫秒,得到 18.207 个 tick,截尾后是 18 个 tick,它实际上是 989 毫秒。对每个小于 55 毫秒的间隔,每个 tick 都会产生一个 WM_TIMER 消息。可见,计时器并不能严格按照指定的时间间隔发送 WM_TIMER 消息,它

17、总要相差那么几毫秒。即使忽略这几个毫秒的差别,计时器仍然不精确。请看原因二:WM_TIMER 消息放在正常的消息队列之中,和其他消息排列在一起,因此,如果在SetTimer 中指定间隔为 1000 毫秒,那么不能保证程序每 1000 毫秒或者 989 毫秒就会收到一个 WM_TIMER 消息。如果其他程序的执行事件超过一秒,在此期间内,您的程式将收不到任何 WM_TIMER 讯息。事实上, Windows 对 WM_TIMER 消息的处理非常类似于对 WM_PAINT消息的处理,这两个消息都是低优先级的,程序只有在消息队列中没有其他消息时才接收它们。WM_TIMER 还在另一方面和 WM_PA

18、INT 相似:Windows 不能持续向消息队列中放入多个WM_TIMER 讯息,而是将多余的 WM_TIMER 消息组合成一个消息。因此,应用程序不会一次收到多个这样的消息,尽管可能在短时间内得到两个 WM_TIMER 消息。应用程序不能确定这种处理方式所导致的 WM_TIMER 消息遗漏的数目。可见,WM_TIMER 消息并不能及时被应用程序所处理,WM_TIMER 在消息队列中的延误可能就不能用毫秒来计算了。由以上两点,你不能通过在处理 WM_TIMER 时一秒一秒计数的方法来计时。如果要实现一个时钟程序,可以使用系统的时间函数如 GetLocalTime ,而在时钟程序中,计时器的作用是定时调用 GetLocalTime 获得新的时间并刷新时钟画面,当然这个刷新的间隔要等于或小于 1 秒。

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 企业管理 > 管理学资料

本站链接:文库   一言   我酷   合作


客服QQ:2549714901微博号:道客多多官方知乎号:道客多多

经营许可证编号: 粤ICP备2021046453号世界地图

道客多多©版权所有2020-2025营业执照举报