多媒體定時器

        雖然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給回調函數傳遞參數


例程如下:

 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}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章