目前,Windows下的定时器编程主要有三种方式。

1)SetTimer定时器是利用Windows窗口消息WM_TIMER来实现的。使用方法非常简单,SetTimer创建定时器,KillTimer销毁定时器。使用条件是调用线程必须要有窗口消息队列message queue,因此如果是工作线程就无法使用这种方法。

2)WaitableTimer定时器,其实应该算是一种线程同步对象,CreateWaitableTimer创建定时器对象,SetWaitableTimer设置定时器回调函数,CLoseHandle销毁定时器。WaitableTimer可以跨线程、进程使用,只要知道定时器对象名字(创建定时器时设置)就可以控制该定时器对象了。WaitableTimer定时器的回调函数实际上是一个APC(Asynchronous Procedure Calls)异步过程调用函数。
关于APC方面的知识可参考“谈谈对APC的一点理解”一文http://blog.csdn.net/wwwwly/archive/2009/07/10/4337907.aspx

3)TimerQueueTimer定时器,应该算迄今为止Windows系统最强大的定时器了。他可以支持多种工作模式,而且定时精度也是最高的。
使用时,首先要调用CreateTimerQueue创建一个定时器队列,然后用CreateTimerQueueTimer来创建一个TimerQueueTimer定时器,
WT_EXECUTEDEFAULT,默认设置,回调函数将进入一个非I/O工作线程队列WT_EXECUTEINTIMERTHREAD,回调函数作为APC,在定时器线程中被调用,被调用的条件是线程进入可警告等待状态alertable wait status。仅适用于短时任务,否则可能会影响队列中的其他定时器。
WT_EXECUTEINIOTHREAD,回调函数进入一个I/O工作线程队列,请注意,大多数定时器都需要调用线程进入可警告等待状态alertable wait status,并不是随随便便就能发生定时调用的。一个线程是否进入可警告等待状态可参见微软的说明http://msdn.microsoft.com/en-us/library/ms686307.aspx
A thread goes into an alertable wait state by calling either SleepEx, MsgWaitForMultipleObjectsEx, WaitForSingleObjectEx, or WaitForMultipleObjectsEx, with the function's bAlertable parameter set to TRUE.
所以希望定时器不受这种可警告等待状态的影响,最好是用TimerQueue来完成。

等待定时器与用户定时器的区别

1、等待定时器(SetWaitableTimer)与用户定时器(SetTimer)它们之间的最大差别是,用户定时器需要在应用程序中设置许多附加的用户界面结构,这使定时器变得资源更加密集。

2、等待定时器属于内核对象,这意味着它们可以供多个线程共享,并且是安全的。

用户定时器:用户定时器能够生成WM_TIMER消息,这些消息将返回给调用SetTimer(用于回调定时器)的线程和创建窗口(用于基于窗口的定时器)的线程。因此,当用户定时器报时的时候,只有一个线程得到通知。

等待定时器:多个线程可以在等待定时器上进行等待,如果定时器是个人工重置的定时器,则可以调度若干个线程。

如果要执行与用户界面相关的事件,以便对定时器作出响应,那么使用用户定时器来组织代码结构可能更加容易些,因为使用等待定时器时,线程必须既要等待各种消息,又要等待内核对象(如果要改变代码的结构,可以使用MsgaitForMultipleObjects函数)。

等待定时器,当到了规定时间的时候,更有可能得到通知。WM_TIMER消息始终属于最低优先级的消息,当线程的队列中没有其他消息时,才检索该消息。等待定时器的处理方法与其他内核对象没有什么差别(PS:因此其使用与内核对象的使用一个样,这就是所谓的触类旁通吧),如果定时器发出报时信息,而你的线程正在等待之中,那么你的线程就会醒来。

一个windows编程示例

#include <iostream>
#include <windows.h>
#include <stdio.h> using namespace std; static int COUNT = ; VOID CALLBACK TimerRoutine(PVOID lpParameter, BOOLEAN TimerOrWaitFired)
{
if (NULL == lpParameter)
{
cout << "TimerRoutine parameter is NULL!" << endl;
return;
} printf("the values of param is :%d\n", *(int*)lpParameter);
} int main(int argc, TCHAR *argv[])
{
HANDLE m_hTimerQueueTimer = NULL;
HANDLE m_hTimerQueue = NULL; while (NULL == (m_hTimerQueue = CreateTimerQueue())) {
cout << "Create timer queue failed!" << endl;
Sleep();
} while (!CreateTimerQueueTimer(&m_hTimerQueueTimer, m_hTimerQueue, WAITORTIMERCALLBACK(TimerRoutine), &COUNT, , , NULL))
{
cout << "Create timer failed!" << endl;
Sleep();
} while () {
Sleep();
COUNT++;
} if (m_hTimerQueueTimer != NULL)
DeleteTimerQueueTimer(m_hTimerQueue, m_hTimerQueueTimer, INVALID_HANDLE_VALUE);
if (m_hTimerQueue != NULL)
DeleteTimerQueueEx(m_hTimerQueue, INVALID_HANDLE_VALUE); m_hTimerQueueTimer = NULL;
m_hTimerQueue = NULL; return ;
}

参考:

1.  Windows系统三种定时器的分析

2. CreateTimerQueueTimer学习笔记

3. Sleep和 SleepEx函数

最新文章

  1. MyEclipse web项目导入Eclipse,详细说明
  2. Java实现深克隆的两种方式
  3. CSS强制英文换行
  4. Android -- Support包特性
  5. 查看堵塞的SQL
  6. VC++ WIN32 sdk实现按钮自绘详解.
  7. 谈谈事件对象-event
  8. echarts自定义提示框数据
  9. Oracle SQL 部分特殊字符转义及escape的用法
  10. Javaweb学习笔记——(十)——————response对象,response字符流缓冲器,响应头,状态码,重定向,requset对象,路径和乱码
  11. 手机端上点击input框软键盘出现时把input框不被覆盖,显示在屏幕中间(转)
  12. 【精】【入门篇】js正则表达式
  13. JS高级 - 面向对象5(继承,引用)
  14. 《Linux内核分析与设计实现》读书笔记一
  15. missing gradle project information
  16. Unity3D学习笔记(四):物理系统碰撞和预制体
  17. EC20 MODULE serial com log in passwd
  18. js基础之DOM中document对象的常用属性方法
  19. linux shell 按行循环读入文件方法
  20. MySQL-事务的实现-redo

热门文章

  1. CF888G XOR-MST trie,贪心
  2. ROS+gre over ipv6,在ipv6的基础上建立GRE隧道
  3. makefile小计
  4. .NET 微服务 学习目录
  5. myeclipse导入项目后中文乱码
  6. 最近C#项目中不小心踩的低级坑
  7. 钉钉与wcp知识库集成工具
  8. 命令(Command)模式
  9. vivado sdx 綜合報錯ERROR: [SdsCompiler 83-5019] Exiting sds++ : Error when calling &#39;aarch64-linux-gnu-g++ -c
  10. Dart方法基础知识