VC知識庫文章 - VC中基于 Windows 的精確定時_第1頁
VC知識庫文章 - VC中基于 Windows 的精確定時_第2頁
VC知識庫文章 - VC中基于 Windows 的精確定時_第3頁
VC知識庫文章 - VC中基于 Windows 的精確定時_第4頁
VC知識庫文章 - VC中基于 Windows 的精確定時_第5頁
已閱讀5頁,還剩2頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領

文檔簡介

1、VC知識庫文章 - VC中基于 Windows 的精確定時 :首頁 >> 文檔中心 >> 在線雜志 >> WIN32 API 在線雜志 第37期 本站特別推薦: Skin+ 通用的界面換膚控件(Skin+ 2.0于2005年11月1日發(fā)布,網(wǎng)站全面更新?。┊a(chǎn)品 特性: 支持多種開發(fā)語言(VC+ ,VB,C#,PowerBuilder,C+Builder,Delphi,E語言等) 界面和業(yè)務邏輯徹底分離,節(jié)約開發(fā)和維護時間,支持各Windows平臺,包括Windows98/NT4/2000/XP/2003 徹底換膚,包括標準控件,通用對話框等,支持對第三方控件

2、的換膚,提供方便的設計工具,可自行設計皮膚。詳情請點這里 原創(chuàng)文檔 本文適合中級讀者 已閱讀23101次 文檔 代碼 工具 VC中基于 Windows 的精確定時 中國科學院光電技術(shù)研究所 游志宇 示例工程下載 在工業(yè)生產(chǎn)控制系統(tǒng)中,有許多需要定時完成的操作,如定時顯示當前時間,定時刷新屏幕上的進度條,上位 機定時向下位機發(fā)送命令和傳送數(shù)據(jù)等。特別是在對控制性能要求較高的實時控制系統(tǒng)和數(shù)據(jù)采集系統(tǒng)中,就更需要精確定時操作。 眾所周知,Windows 是基于消息機制的系統(tǒng),任何事件的執(zhí)行都是通過發(fā)送和接收消息來完成的。 這樣就帶來了一些問題,如一旦計算機的CPU被某個進程占用,或系統(tǒng)資源緊張時,

3、發(fā)送到消息隊列 中的消息就暫時被掛起,得不到實時處理。因此,不能簡單地通過Windows消息引發(fā)一個對定時要求 嚴格的事件。另外,由于在Windows中已經(jīng)封裝了計算機底層硬件的訪問,所以,要想通過直接利用 訪問硬件來完成精確定時,也比較困難。所以在實際應用時,應針對具體定時精度的要求,采取相適 應的定時方法。 VC中提供了很多關(guān)于時間操作的函數(shù),利用它們控制程序能夠精確地完成定時和計時操作。本文詳細介紹了 VC中基于Windows的精確定時的七種方式,如下圖所示: 圖一 圖像描述 方式一:VC中的WM_TIMER消息映射能進行簡單的時間控制。首先調(diào)用函數(shù)SetTimer()設置定時 間隔,如

4、SetTimer(0,200,NULL)即為設置200ms的時間間隔。然后在應用程序中增加定時響應函數(shù) OnTimer(),并在該函數(shù)中添加響應的處理語句,用來完成到達定時時間的操作。這種定時方法非常 簡單,可以實現(xiàn)一定的定時功能,但其定時功能如同Sleep()函數(shù)的延時功能一樣,精度非常低,最小 計時精度僅為30ms,CPU占用低,且定時器消息在多任務操作系統(tǒng)中的優(yōu)先級很低,不能得到及時響 應,往往不能滿足實時控制環(huán)境下的應用。只可以用來實現(xiàn)諸如位圖的動態(tài)顯示等對定時精度要求不高的情況。如示例工程中的Timer1。 方式二:VC中使用sleep()函數(shù)實現(xiàn)延時,它的單位是ms,如延時2秒,用

5、sleep(2000)。精度非常 低,最小計時精度僅為30ms,用sleep函數(shù)的不利處在于延時期間不能處理其他的消息,如果時間太 長,就好象死機一樣,CPU占用率非常高,只能用于要求不高的延時程序中。如示例工程中的Timer2。 方式三:利用COleDateTime類和COleDateTimeSpan類結(jié)合WINDOWS的消息處理過程來實現(xiàn)秒級延時。如示例工程中的Timer3和Timer3_1。以下是實現(xiàn)2秒的延時代碼: COleDateTime start_time = COleDateTime:GetCurrentTime(); COleDateTimeSpan end_time= CO

6、leDateTime:GetCurrentTime()-start_time; while(end_time.GetTotalSeconds()< 2) /實現(xiàn)延時2秒 MSG msg; GetMessage(&msg,NULL,0,0); TranslateMessage(&msg); DispatchMessage(&msg); /以上四行是實現(xiàn)在延時或定時期間能處理其他的消息, /雖然這樣可以降低CPU的占有率, /但降低了延時或定時精度,實際應用中可以去掉。 end_time = COleDateTime:GetCurrentTime()-start_ti

7、me; /這樣在延時的時候我們也能夠處理其他的消息。 方式四:在精度要求較高的情況下,VC中可以利用GetTickCount()函數(shù),該函數(shù)的返回值是 DWORD型,表示以ms為單位的計算機啟動后經(jīng)歷的時間間隔。精度比WM_TIMER消息映射高,在較 短的定時中其計時誤差為15ms,在較長的定時中其計時誤差較低,如果定時時間太長,就好象死機一樣,CPU占用率非常高,只能用于要求不高的延時程序中。如示例工程中的Timer4和Timer4_1。下列代碼可以實現(xiàn)50ms的精確定時: DWORD dwStart = GetTickCount(); DWORD dwEnd = dwStart; do d

8、wEnd = GetTickCount()-dwStart; while(dwEnd <50);為使GetTickCount()函數(shù)在延時或定時期間能處理其他的消息,可以把代碼改為: DWORD dwStart = GetTickCount(); DWORD dwEnd = dwStart; do MSG msg; GetMessage(&msg,NULL,0,0); TranslateMessage(&msg); DispatchMessage(&msg); dwEnd = GetTickCount()-dwStart; while(dwEnd <50);

9、雖然這樣可以降低CPU的占有率,并在延時或定時期間也能處理其他的消息,但降低了延時或定時精度。 方式五:與GetTickCount()函數(shù)類似的多媒體定時器函數(shù)DWORD timeGetTime(void),該函數(shù)定時精 度為ms級,返回從Windows啟動開始經(jīng)過的毫秒數(shù)。微軟公司在其多媒體Windows中提供了精確定時器的底 層API持,利用多媒體定時器可以很精確地讀出系統(tǒng)的當前時間,并且能在非常精確的時間間隔內(nèi)完成一 個事件、函數(shù)或過程的調(diào)用。不同之處在于調(diào)用DWORD timeGetTime(void) 函數(shù)之前必須將 Winmm.lib 和 Mmsystem.h 添加到工程中,否則在

10、編譯時提示DWORD timeGetTime(void)函數(shù)未定義。由于使用該 函數(shù)是通過查詢的方式進行定時控制的,所以,應該建立定時循環(huán)來進行定時事件的控制。如示例工程中的Timer5和Timer5_1。 方式六:使用多媒體定時器timeSetEvent()函數(shù),該函數(shù)定時精度為ms級。利用該函數(shù)可以實現(xiàn)周期性的函數(shù)調(diào)用。如示例工程中的Timer6和Timer6_1。函數(shù)的原型如下: MMRESULT timeSetEvent( UINT uDelay, UINT uResolution, LPTIMECALLBACK lpTimeProc, WORD dwUser, UINT fuEven

11、t )該函數(shù)設置一個定時回調(diào)事件,此事件可以是一個一次性事件或周期性事件。事件一旦被激活,便調(diào)用指定的回調(diào)函數(shù), 成功后返回事件的標識符代碼,否則返回NULL。函數(shù)的參數(shù)說明如下: uDelay:以毫秒指定事件的周期。 Uresolution:以毫秒指定延時的精度,數(shù)值越小定時器事件分辨率越高。缺省值為1ms。 LpTimeProc:指向一個回調(diào)函數(shù)。 DwUser:存放用戶提供的回調(diào)數(shù)據(jù)。 FuEvent:指定定時器事件類型: TIME_ONESHOT:uDelay毫秒后只產(chǎn)生一次事件 TIME_PERIODIC :每隔uDelay毫秒周期性地產(chǎn)生事件。 具體應用時,可以通過調(diào)用timeSe

12、tEvent()函數(shù),將需要周期性執(zhí)行的任務定義在LpTimeProc回調(diào)函數(shù) 中(如:定時采樣、控制等),從而完成所需處理的事件。需要注意的是,任務處理的時間不能大于周期間隔時間。另外,在定時器使用完畢后, 應及時調(diào)用timeKillEvent()將之釋放。 方式七:對于精確度要求更高的定時操作,則應該使用QueryPerformanceFrequency()和 QueryPerformanceCounter()函數(shù)。這兩個函數(shù)是VC提供的僅供Windows 95及其后續(xù)版本使用的精確時間函數(shù),并要求計算機從硬件上支持精確定時器。如示例工程中的Timer7、Timer7_1、Timer7_2

13、、Timer7_3。 QueryPerformanceFrequency()函數(shù)和QueryPerformanceCounter()函數(shù)的原型如下: BOOL QueryPerformanceFrequency(LARGE_INTEGER lpFrequency); BOOL QueryPerformanceCounter(LARGE_INTEGER lpCount);數(shù)據(jù)類型ARGE_INTEGER既可以是一個8字節(jié)長的整型數(shù),也可以是兩個4字節(jié)長的整型數(shù)的聯(lián)合結(jié)構(gòu), 其具體用法根據(jù)編譯器是否支持64位而定。該類型的定義如下: typedef union _LARGE_INTEGER str

14、uct DWORD LowPart ;/ 4字節(jié)整型數(shù) LONG HighPart;/ 4字節(jié)整型數(shù) ; LONGLONG QuadPart ;/ 8字節(jié)整型數(shù) LARGE_INTEGER ;在進行定時之前,先調(diào)用QueryPerformanceFrequency()函數(shù)獲得機器內(nèi)部定時器的時鐘頻率, 然后在需要嚴格定時的事件發(fā)生之前和發(fā)生之后分別調(diào)用QueryPerformanceCounter()函數(shù),利用兩次獲得的計數(shù)之差及時鐘頻率,計算出事件經(jīng) 歷的精確時間。下列代碼實現(xiàn)1ms的精確定時: LARGE_INTEGER litmp; LONGLONG QPart1,QPart2; dou

15、ble dfMinus, dfFreq, dfTim; QueryPerformanceFrequency(&litmp); dfFreq = (double)litmp.QuadPart;/ 獲得計數(shù)器的時鐘頻率 QueryPerformanceCounter(&litmp); QPart1 = litmp.QuadPart;/ 獲得初始值 do QueryPerformanceCounter(&litmp); QPart2 = litmp.QuadPart;/獲得中止值 dfMinus = (double)(QPart2-QPart1); dfTim = dfMin

16、us / dfFreq;/ 獲得對應的時間值,單位為秒 while(dfTim<0.001);其定時誤差不超過1微秒,精度與CPU等機器配置有關(guān)。 下面的程序用來測試函數(shù)Sleep(100)的精確持續(xù)時間: LARGE_INTEGER litmp; LONGLONG QPart1,QPart2; double dfMinus, dfFreq, dfTim; QueryPerformanceFrequency(&litmp); dfFreq = (double)litmp.QuadPart;/ 獲得計數(shù)器的時鐘頻率 QueryPerformanceCounter(&litm

17、p); QPart1 = litmp.QuadPart;/ 獲得初始值 Sleep(100); QueryPerformanceCounter(&litmp); QPart2 = litmp.QuadPart;/獲得中止值 dfMinus = (double)(QPart2-QPart1); dfTim = dfMinus / dfFreq;/ 獲得對應的時間值,單位為秒 由于Sleep()函數(shù)自身的誤差,上述程序每次執(zhí)行的結(jié)果都會有微小誤差。下列代碼實現(xiàn)1微秒的精確定時: LARGE_INTEGER litmp; LONGLONG QPart1,QPart2; double dfMi

18、nus, dfFreq, dfTim; QueryPerformanceFrequency(&litmp); dfFreq = (double)litmp.QuadPart;/ 獲得計數(shù)器的時鐘頻率 QueryPerformanceCounter(&litmp); QPart1 = litmp.QuadPart;/ 獲得初始值 do QueryPerformanceCounter(&litmp); QPart2 = litmp.QuadPart;/獲得中止值 dfMinus = (double)(QPart2-QPart1); dfTim = dfMinus / dfF

19、req;/ 獲得對應的時間值,單位為秒 while(dfTim<0.000001);其定時誤差一般不超過0.5微秒,精度與CPU等機器配置有關(guān)。(完) 最新評論 發(fā)表評論 文章投稿 查看所有評論 推薦給好友 打印 你好,我是數(shù)據(jù)庫編程的初學者,請問當點擊一個按鈕進入下一頁面的時候有時總是會出現(xiàn)一個錯誤:提示指針錯誤或者檢查記錄出錯,但還是能進入下一頁面,請問是為什么? 還有就是無效的游標狀態(tài)又是怎么回事? ( 153375424 發(fā)表于 2005-11-28 0:18:00) 你好,我是數(shù)據(jù)庫編程的初學者,請問當點擊一個按鈕進入下一頁面的時候有時總是會出現(xiàn)一個錯誤:提示指針錯誤或者檢查記

20、錄出錯,但還是能進入下一頁面,請問是為什么? 還有就是無效的游標狀態(tài)又是怎么回事? ( 153375424 發(fā)表于 2005-11-28 0:18:00) 不錯,很全面 ( hkbs_1121 發(fā)表于 2005-3-31 13:44:00) 寫得很好.謝謝. ( ot512 發(fā)表于 2004-12-6 12:12:00) 以下是個人想法: 首先,在應用程序級做精確定時是相當艱難的。 這里面很多因素要考慮:比如多任務操作系統(tǒng)固有的任務切換;其他驅(qū)動程序的CPU時間的搶占;高優(yōu)先級應用程序的執(zhí)行;不確定的操作系統(tǒng)的作業(yè)任務分配規(guī)則(其實低優(yōu)先級的線程也可以獲得運行)等等許多問題,都可以導致定時的不精確。 其實,即使在驅(qū)動程序級別要想做到精確定時也非易事,這里面還有定時間隔的準確度和精度問題。直接采用計算機的原有硬件,由于計算機的個體差異,會導致定時精度在總體上難以保障。所以,要提高精度,許多人還是采用了專用外圍定時集成電路的方法。通過驅(qū)動程序和硬件相結(jié)合,以達到高精度

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
  • 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論