雖然Win95下可視化開發工具如VC、Delphi、C++ Builder等都有專用的定時器控件Timer,而且使用很方便,可以實現一定的定時功能,但最小計時精度僅為55ms,且定時器消息在多任務操作系統中的優先級很低,不能得到及時響應,往往不能滿足實時控制環境下的應用。不過Microsoft公司在Win32 API函數庫中已經為用戶提供了一組用于高精度計時的底層函數,如果用戶使用得當,
計時精度可到1ms。這個計時精度、對于一般的實時系統控制完全可以滿足要求。現將由C++ Builder 4.0提供的重新封裝后的一組與時間相關的主要接口函數(函數名、參數、功能與Win32 API基本相同)說明如下:
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給回調函數傳遞參數例程如下:
1
MMRESULT g_wTimerID = 0;
//回調函數,參數不能有錯
2
void CALLBACK CDsisiiDlg::SendFun(UINT wTimerID, UINT msg, DWORD dwUser, DWORD dwl, DWORD dw2)
3

{
CDsisiiDlg* pdcpackerdlg = (CDsisiiDlg*)dwUser;
...
4
}
5
6
bool 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
//刪除定時器
30
void CDsisiiDlg::DestroyTimer()
31

{
32
if (g_wTimerID)
33
{
34
timeKillEvent(g_wTimerID);
35
g_wTimerID = 0;
36
}
37
}
posted on 2009-09-12 13:53
Bluesea 閱讀(3734)
評論(0) 編輯 收藏 引用 所屬分類:
C/C++