VC中基于 Windows 的精確定時
示例工程下載
在工業生產控制系統中,有許多需要定時完成的操作,如定時顯示當前時間,定時刷新屏幕上的進度條,上位
眾所周知,Windows 是基于消息機制的系統,任何事件的執行都是通過發送和接收消息來完成的。
VC中提供了很多關于時間操作的函數,利用它們控制程序能夠精確地完成定時和計時操作。本文詳細介紹了
圖一 圖像描述
方式一:VC中的WM_TIMER消息映射能進行簡單的時間控制。首先調用函數SetTimer()設置定時
方式二:VC中使用sleep()函數實現延時,它的單位是ms,如延時2秒,用sleep(2000)。精度非常
方式三:利用COleDateTime類和COleDateTimeSpan類結合WINDOWS的消息處理過程來實現秒級延時。如示例工程中的Timer3和Timer3_1。以下是實現2秒的延時代碼:
COleDateTime start_time = COleDateTime::GetCurrentTime(); COleDateTimeSpan end_time= COleDateTime::GetCurrentTime()-start_time; while(end_time.GetTotalSeconds()< 2) //實現延時2秒 { MSG msg; GetMessage(&msg,NULL,0,0); TranslateMessage(&msg); DispatchMessage(&msg); //以上四行是實現在延時或定時期間能處理其他的消息, //雖然這樣可以降低CPU的占有率, //但降低了延時或定時精度,實際應用中可以去掉。 end_time = COleDateTime::GetCurrentTime()-start_time; }//這樣在延時的時候我們也能夠處理其他的消息。方式四:在精度要求較高的情況下,VC中可以利用GetTickCount()函數,該函數的返回值是
DWORD dwStart = GetTickCount(); DWORD dwEnd = dwStart; do { dwEnd = GetTickCount()-dwStart; }while(dwEnd <50);為使GetTickCount()函數在延時或定時期間能處理其他的消息,可以把代碼改為:
DWORD dwStart = GetTickCount(); DWORD dwEnd = dwStart; do { MSG msg; GetMessage(&msg,NULL,0,0); TranslateMessage(&msg); DispatchMessage(&msg); dwEnd = GetTickCount()-dwStart; }while(dwEnd <50);雖然這樣可以降低CPU的占有率,并在延時或定時期間也能處理其他的消息,但降低了延時或定時精度。
方式五:與GetTickCount()函數類似的多媒體定時器函數DWORD timeGetTime(void),該函數定時精
方式六:使用多媒體定時器timeSetEvent()函數,該函數定時精度為ms級。利用該函數可以實現周期性的函數調用。如示例工程中的Timer6和Timer6_1。函數的原型如下:
MMRESULT timeSetEvent( UINT uDelay, UINT uResolution, LPTIMECALLBACK lpTimeProc, WORD dwUser, UINT fuEvent )該函數設置一個定時回調事件,此事件可以是一個一次性事件或周期性事件。事件一旦被激活,便調用指定的回調函數,
uDelay:以毫秒指定事件的周期。 Uresolution:以毫秒指定延時的精度,數值越小定時器事件分辨率越高。缺省值為1ms。 LpTimeProc:指向一個回調函數。 DwUser:存放用戶提供的回調數據。 FuEvent:指定定時器事件類型: TIME_ONESHOT:uDelay毫秒后只產生一次事件 TIME_PERIODIC :每隔uDelay毫秒周期性地產生事件。具體應用時,可以通過調用timeSetEvent()函數,將需要周期性執行的任務定義在LpTimeProc回調函數
方式七:對于精確度要求更高的定時操作,則應該使用QueryPerformanceFrequency()和
QueryPerformanceFrequency()函數和QueryPerformanceCounter()函數的原型如下:
BOOL QueryPerformanceFrequency(LARGE_INTEGER *lpFrequency); BOOL QueryPerformanceCounter(LARGE_INTEGER *lpCount);數據類型ARGE_INTEGER既可以是一個8字節長的整型數,也可以是兩個4字節長的整型數的聯合結構,
typedef union _LARGE_INTEGER { struct { DWORD LowPart ;// 4字節整型數 LONG HighPart;// 4字節整型數 }; LONGLONG QuadPart ;// 8字節整型數 }LARGE_INTEGER ;在進行定時之前,先調用QueryPerformanceFrequency()函數獲得機器內部定時器的時鐘頻率,
LARGE_INTEGER litmp; LONGLONG QPart1,QPart2; double dfMinus, dfFreq, dfTim; QueryPerformanceFrequency(&litmp); dfFreq = (double)litmp.QuadPart;// 獲得計數器的時鐘頻率 QueryPerformanceCounter(&litmp); QPart1 = litmp.QuadPart;// 獲得初始值 do { QueryPerformanceCounter(&litmp); QPart2 = litmp.QuadPart;//獲得中止值 dfMinus = (double)(QPart2-QPart1); dfTim = dfMinus / dfFreq;// 獲得對應的時間值,單位為秒 }while(dfTim<0.001);其定時誤差不超過1微秒,精度與CPU等機器配置有關。 下面的程序用來測試函數Sleep(100)的精確持續時間:
LARGE_INTEGER litmp; LONGLONG QPart1,QPart2; double dfMinus, dfFreq, dfTim; QueryPerformanceFrequency(&litmp); dfFreq = (double)litmp.QuadPart;// 獲得計數器的時鐘頻率 QueryPerformanceCounter(&litmp); QPart1 = litmp.QuadPart;// 獲得初始值 Sleep(100); QueryPerformanceCounter(&litmp); QPart2 = litmp.QuadPart;//獲得中止值 dfMinus = (double)(QPart2-QPart1); dfTim = dfMinus / dfFreq;// 獲得對應的時間值,單位為秒由于Sleep()函數自身的誤差,上述程序每次執行的結果都會有微小誤差。下列代碼實現1微秒的精確定時:
LARGE_INTEGER litmp; LONGLONG QPart1,QPart2; double dfMinus, dfFreq, dfTim; QueryPerformanceFrequency(&litmp); dfFreq = (double)litmp.QuadPart;// 獲得計數器的時鐘頻率 QueryPerformanceCounter(&litmp); QPart1 = litmp.QuadPart;// 獲得初始值 do { QueryPerformanceCounter(&litmp); QPart2 = litmp.QuadPart;//獲得中止值 dfMinus = (double)(QPart2-QPart1); dfTim = dfMinus / dfFreq;// 獲得對應的時間值,單位為秒 }while(dfTim<0.000001);其定時誤差一般不超過0.5微秒,精度與CPU等機器配置有關。(完)
from: http://www.vckbase.com/document/viewdoc/?id=1301
posted on 2006-11-11 12:57 weidagang2046 閱讀(396) 評論(0) 編輯 收藏 所屬分類: Windows