基本概念

SetTimer函数用于创建一个计时器

KillTimer函数用于销毁一个计时器。

计时器属于系统资源,使用完应及时销毁。

API

SetTimer的函数原型:

UINT_PTR SetTimer( HWND hWnd, UINT_PTR nIDEvent, UINT uElapse, TIMERPROC lpTimerFunc ) ; 

其中

hWnd是和timer关联的窗口句柄,此窗口必须为调用SetTimer的线程所有;如果hWnd为NULL,没有窗口和timer相关联并且nIDEvent参数被忽略

nIDEvent是timer的标识,为非零值;如果hWnd为NULL则被忽略;如果hWnd非NULL而且与timer相关联的窗口已经存在一个为此标识的timer,则此次SetTimer调用将用新的timer代替原来的timer。timer标识和窗口相关,两个不同的窗口可以拥有nIDEvent相同的tiemr

uElapse是以毫秒指定的计时间隔值,范围为1毫秒到4,294,967,295毫秒(将近50天),这个值指示Windows每隔多久时间给程序发送WM_TIMER消息。

lpTimerFunc是一个回调函数的指针,俗称TimerFunc;如果lpTimerFunc为NULL,系统将向应用程序队列发送WM_TIMER消息;如果lpTimerFunc指定了一个值,DefWindowProc将在处理WM_TIMER消息时调用这个lpTimerFunc所指向的回调函数,因此即使使用TimerProc代替处理WM_TIMER也需要向窗口分发消息。

返回值:如果hWnd为NULL,返回值为新建立的timer的ID,如果hWnd非NULL,返回一个非0整数,如果SetTimer调用失败则返回0

KillTimer的函数原型:

BOOL KillTimer( HWND hWnd, UINT_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 messages. 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的指针,如果调用SetTimer时没有指定TimerProc(参数值为NULL),则lParam为0(即NULL)。 
可以通过在窗口过程中提供一个WM_TIMER case处理这个消息,或者,默认窗口过程会调用SetTimer中指定的TimerProc来处理WM_TIMER消息

使用方法

如果在程序的整个执行过程中使用计时器,一般在处理WM_CREATE消息时或WinMain中消息循环前调用SetTimer,在处理WM_DESTROY消息时或在WinMain中消息循环后return前调用KillTimer。根据SetTimer中的参数不同,有三种方法使用计时器。

方法一

调用SetTimer时指定窗口句柄hWnd,nIDEvent中指定计时器ID,将lpTimerFunc置NULL从而不使用TimerProc;

在窗口过程中处理WM_TIMER消息。

调用KillTimer时,使用SetTimer中指定的hWnd和id。

最好使用#define定义timer的id,例如:

#define ID_TIMER 1
SetTimer(hWnd,ID_TIMER,1000,NULL) ;
KillTimer(hWnd,ID_TIMER) ; 

方法二

调用SetTimer时指定窗口句柄hWnd,nIDEvent中指定计时器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值是计时器ID,dwTimer值是与从GetTickCount函数的返回值相容的值。这是自Windows启动后所经过的毫秒数。

使用这种方法时,相关函数调用的形式为:

SetTimer(hWnd,ID_TIMER,1000,TimerProc) ;
KillTimer(hWnd,ID_TIMER) ; 

方法三

调用SetTimer时不指定窗口句柄(为NULL),iTimerID参数自然被忽略,lpTimerFunc不为NULL而指定为TimerProc的指针。正如上面SetTimer的讨论中所说的,此时SetTimer的返回值正是新建立的计时器的ID,需将这个ID保存以供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
然后使用两个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。

常见问题

计时器精确吗?

计时器并不精确。有两个原因:

原因一: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 Windows 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消息,它总要相差那么几毫秒。

即使忽略这几个毫秒的差别,计时器仍然不精确。请看原因二:

WM_TIMER消息放在正常的消息队列之中,和其他消息排列在一起,因此,如果在SetTimer中指定间隔为1000毫秒,那么不能保证程序每1000毫秒或者989毫秒就会收到一个WM_TIMER消息。如果其他程序的执行事件超过一秒,在此期间内,您的程式将收不到任何WM_TIMER讯息。事实上, Windows对WM_TIMER消息的处理非常类似于对WM_PAINT消息的处理,这两个消息都是低优先级的,程序只有在消息队列中没有其他消息时才接收它们。 
WM_TIMER还在另一方面和WM_PAINT相似:Windows不能持续向消息队列中放入多个WM_TIMER讯息,而是将多余的WM_TIMER消息组合成一个消息。因此,应用程序不会一次收到多个这样的消息,尽管可能在短时间内得到两个WM_TIMER消息。应用程序不能确定这种处理方式所导致的WM_TIMER消息「遗漏」的数目。 
可见,WM_TIMER消息并不能及时被应用程序所处理,WM_TIMER在消息队列中的延误可能就不能用毫秒来计算了。

由以上两点,你不能通过在处理WM_TIMER时一秒一秒计数的方法来计时。如果要实现一个时钟程序,可以使用系统的时间函数如GetLocalTime ,而在时钟程序中,计时器的作用是定时调用GetLocalTime获得新的时间并刷新时钟画面,当然这个刷新的间隔要等于或小于1秒。

WM_TIMER消息无法响应

https://bbs.csdn.net/topics/300035118?list=lz

参考文章

http://www.studynew.com/study/vc/2004082519380331684.html

http://blog.sina.com.cn/s/blog_5a6f39cf010148xy.html

https://blog.csdn.net/u012104827/article/details/105278079/

https://blog.csdn.net/lcalqf/article/details/53020009

Visual C++——定时器(计时器)SetTimer函数相关推荐

  1. vc中定时器的使用 (2)SetTimer函数的使用

    什么时候我们需要用到SetTimer函数呢?当你需要每个一段时间执行一件事的的时候就需要使用SetTimer函数了. 使用定时器的方法比较简单,通常告诉WINDOWS一个时间间隔,然后WINDOWS以 ...

  2. VC++中定时器SetTimer函数的应用总结

    什么时候我们需要用到SetTimer函数呢?当你 需要每个一段时间执行一件事的的时候就需要使用SetTimer函数了.使用定时器的方法比较简单,通常告诉WINDOWS一个时间间隔,然后WINDOWS以 ...

  3. 定时器:SetTimer

    注:把编辑框属性设置为"只读" 什么时候我们需要用到SetTimer函数呢?当你需要"每个一段时间"执行一件事的的时候就需要使用SetTimer函数了.使用定时 ...

  4. MFC 定时器用法 SetTimer、KillTimer

    什么时候我们需要用到SetTimer函数呢?当你需要每个一段时间执行一件事的的时候就需要使用SetTimer函数了. 使用定时器的方法比较简单,通常告诉WINDOWS一个时间间隔,然后WINDOWS以 ...

  5. windows中定时操作(SetTimer函数用法)

    最近需要在服务端定时执行某个操作,于是需要用到SetTimer函数.当你需要每个一段时间执行一件事的的时候就需要使用SetTimer函数了. 使用定时器的方法比较简单,通常告诉WINDOWS一个时间间 ...

  6. delphi中setTimer函数的用法

    delphisetTimer函数用来做定时器,可以实现每隔一段时间运行某个程序,需要用到两个函数,SetTimer函数和KillTimer函数. 下面是一个简单的例子: var i :Integer ...

  7. javascript基础复习之函数,定时器,erval函数

    Function对象 JS中函数就是Function对象 函数名就是指向Function对象的引用 使用函数名就可以访问函数的对象 函数名()是调用函数 function 函数名 ([参数]) {函数 ...

  8. 51单片机 AT89C52 的中断/定时器/计时器 笔记

    第二次学习 听课笔记 中断允许寄存器IE(地址为A8H,可位寻址) EA:总中断允许控制位: 2外部中断+3内部定时器中断/计时器中断+1串口中断 EX0  EX1  外部中断: ET0  ET1  ...

  9. 制作秒表——AT24C02储存芯片与用定时器代替Delay函数

    本来准备学linux的,但临时决定先把单片机的基础知识学完. 本周学习内容: 1.学到了一个很好的用定时器代替Delay函数的方法.利用多个累加变量可在一个定时器中断函数中满足多个功能的延时需求. 2 ...

  10. Visual foxpro 命令大全和函数大全

    Visual foxpro 命令大全和函数大全 ? 在下一行显示表达式串 ?? 在当前行显示表达式串 @... 将数据按用户设定的格式显示在屏幕上或在打印机上打印 ACCEPT 把一个字符 ...

最新文章

  1. 32个程序员萌翻全场的瞬间!
  2. kali2018安装教程_Kali Linux + Windows10双系统安装教程
  3. 被“轻视”的CV·AR的背后核心技术
  4. win7打不开qq文件怎么办
  5. [vijos1982][NOIP2015]子串
  6. 启动tomcat控制台日志出现乱码怎么办?
  7. 夏令时 DST (Daylight Saving Time) java中的夏令时。
  8. 数据冲突Statspack ORA-00001 unique constraint violated错误的解决
  9. linux set 39 date 39,Linux date命令
  10. FireBug不能用了?使用火狐Try Xpath插件替代Firebug和Firepath
  11. C++ Eigen库实现矩阵的CholeSky分解
  12. 第一讲 ISO 17799/27001 标准简介
  13. 网上流行护眼色的RGB值和颜色代码汇总
  14. 鸿蒙系统安全模式,菜鸟必看 如何在安全模式下使用光驱
  15. 企业文档管理难题,云盒子来解决!
  16. ARM+SD2405 IIC_RTC驱动编写及IIC通讯协议
  17. PHP全站pjax影响收录,zblogPHP增加pjax功能,大写的一个“帅”字 - 胡言乱语
  18. 若是所有散户都不去打新会怎样?
  19. 数据结构 课程设计报告
  20. 证券市场基础知识(一)——证券市场概述

热门文章

  1. Java反编译工具-luyten
  2. 孪生网络图像相似度_Siamese network 孪生神经网络一个简单神奇的结构
  3. 手机音频拼接软件_介绍三款无损分割视频软件
  4. 双非一本考研湖南大学计算机技术初试经验分享
  5. CentOS向日葵安装
  6. VCPKG 包下载失败解决思路
  7. DB2 SQLCODE 异常大全编辑(二)
  8. Ansible+Redfish+Idrac管理DELL服务器
  9. iPhone app 和android app开发环境和语言的区别
  10. LBP特征提取(C++实现)