1.DWORD timeGetTime(void)
返回從Windows啓動開始經過的毫秒數。最大值爲232,約49.71天。
2.MMRESULT timeSetEvent(
UINT uDelay,
UINT uResolution,
LPTIMECALLBACK lpTimeProc,
DWORD dwUser,
UINT fuEvent)
該函數設置一個定時回調事件,此事件可以是一個一次性事件或週期性事件。事件一旦被激活,便調用指定的回調函數,成功後返回事件的標識符代碼,否則返回NULL。參數說明如下:
uDelay:以毫秒指定事件的週期。
UResolution:以毫秒指定延時的精度,數值越小定時器事件分辨率越高。缺省值爲1ms。
LpTimeProc:指向一個回調函數。
DwUser:存放用戶提供的回調數據。
FuEvent:指定定時器事件類型:
TIME_ONESHOT:uDelay毫秒後只產生一次事件
TIME_PERIODIC :每隔uDelay毫秒週期性地產生事件。
3.MMRESULT timeKillEvent(UINT uTimerID)
該函數取消一個指定的定時器回調事件。uTimerID標識要取消的事件(由timeSetEvent函數返回的標識符)。如果成功則返回TIMERR_NOERROR,如果定時器時間不存在則返回MMSYSERR_INVALPARAM。
4.回調函數
void CALLBACK TimeProc(
UINT uID,
UINT uMsg,
DWORD dwUser,
DWORD dw1,
DWORD dw2);
該函數是一個應用程序定義的回調函數,出現定時器事件時該函數被調用。TimeProc是應用程序定義的函數名的佔位符。使用該函數
時要注意的是,它只能調用以下有限的幾組API函數:PostMessage,timeGetSystemTime, timeGetTime, timeSetEvent,timeKillEvent
,midiOutShortMsg, midiOutLongMsg,OutputDebugString。同時也不要使用完成時間很長的API函數,程序儘可能簡短。
使用以上一組函數就可以完成毫秒級精度的計時和控制(在C++Builder中使用時要將頭文件mmsystem.h加到程序中)。由於將定時控
制精確到幾毫秒,定時器事件將佔用大量的CPU時間和系統資源,所以在滿足控制要求的前提下,應儘量將參數uResolution的數值增大。而
且定時器實時控制功能完成後要儘快釋放。
注意以下幾點問題:
一、回調函數的參數不能有誤,否則可能引起程序崩掉;
二、事件調用週期uDelay不能小於事件處理時間,否則會引起程序崩潰;
三、通過dwUser給回調函數傳遞參數
例程如下:
1MMRESULT g_wTimerID = 0;
//回調函數,參數不能有錯
2void CALLBACK CDsisiiDlg::SendFun(UINT wTimerID, UINT msg, DWORD dwUser, DWORD dwl, DWORD dw2)
3{
CDsisiiDlg* pdcpackerdlg = (CDsisiiDlg*)dwUser;
...
4}
5
6bool CDsisiiDlg::CreateTimer()
7{
8 TIMECAPS tc;
9 UINT wTimerRes;
10
11 //設置多媒體定時器
12 if(timeGetDevCaps(&tc,sizeof(TIMECAPS))!=TIMERR_NOERROR)//向機器申請一個多媒體定時器
13 return false;
14
15 //獲得機器允許的時間間隔(一般可達到1毫秒)
16 wTimerRes=min(max(tc.wPeriodMin,1),tc.wPeriodMax);
17
18 //定時器開始工作
19 timeBeginPeriod(wTimerRes);
20
21 //每過6毫秒調用回調函數timerback(),wTimerID爲定時器ID.TIME_PERIODIC表週期性調用,TIME_ONESHOT表只產生一次事件
22 g_wTimerID = timeSetEvent(6, wTimerRes, (LPTIMECALLBACK)SendFun, (DWORD)this, TIME_PERIODIC);
23 if(g_wTimerID == 0)
24 return false;
25
26 return true;
27}
28
29//刪除定時器
30void CDsisiiDlg::DestroyTimer()
31{
32 if (g_wTimerID)
33 {
34 timeKillEvent(g_wTimerID);
35 g_wTimerID = 0;
36 }
37}
//回調函數,參數不能有錯
2void CALLBACK CDsisiiDlg::SendFun(UINT wTimerID, UINT msg, DWORD dwUser, DWORD dwl, DWORD dw2)
3{
CDsisiiDlg* pdcpackerdlg = (CDsisiiDlg*)dwUser;
...
4}
5
6bool CDsisiiDlg::CreateTimer()
7{
8 TIMECAPS tc;
9 UINT wTimerRes;
10
11 //設置多媒體定時器
12 if(timeGetDevCaps(&tc,sizeof(TIMECAPS))!=TIMERR_NOERROR)//向機器申請一個多媒體定時器
13 return false;
14
15 //獲得機器允許的時間間隔(一般可達到1毫秒)
16 wTimerRes=min(max(tc.wPeriodMin,1),tc.wPeriodMax);
17
18 //定時器開始工作
19 timeBeginPeriod(wTimerRes);
20
21 //每過6毫秒調用回調函數timerback(),wTimerID爲定時器ID.TIME_PERIODIC表週期性調用,TIME_ONESHOT表只產生一次事件
22 g_wTimerID = timeSetEvent(6, wTimerRes, (LPTIMECALLBACK)SendFun, (DWORD)this, TIME_PERIODIC);
23 if(g_wTimerID == 0)
24 return false;
25
26 return true;
27}
28
29//刪除定時器
30void CDsisiiDlg::DestroyTimer()
31{
32 if (g_wTimerID)
33 {
34 timeKillEvent(g_wTimerID);
35 g_wTimerID = 0;
36 }
37}