Windows系統三種定時器的分析
目前,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來完成。
目前,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來完成。


