總結(jié)了一些關(guān)于多媒體定時(shí)器的使用和處理跨線程更新窗口的原理和方法
微軟在32位版本的系統(tǒng)里提供了一組所謂的"多媒體定時(shí)器"API,多媒體定時(shí)器可以使應(yīng)用程序最大限度的獲得硬件平臺(tái)支持的定時(shí)精度。可以實(shí)現(xiàn)高精度的定時(shí),例如可以應(yīng)用于 MIDI序列發(fā)生器,MIDI時(shí)間產(chǎn)生的精度在一毫秒之內(nèi)。
一、多媒體定時(shí)器的使用方法
設(shè)置多媒體定時(shí)器timeSetEvent()函數(shù),定時(shí)精度為ms級(jí)。利用該函數(shù)可以實(shí)現(xiàn)周期性的函數(shù)調(diào)用。
1、函數(shù)的原型如下:
MMRESULT timeSetEvent( UINT uDelay,
UINT uResolution,
LPTIMECALLBACK lpTimeProc,
WORD dwUser,
UINT fuEvent )
該函數(shù)設(shè)置一個(gè)定時(shí)回調(diào)事件,此事件可以是一個(gè)一次性事件或周期性事件。事件一旦被激活,便調(diào)用指定的回調(diào)函數(shù), 成功后返回事件的標(biāo)識(shí)符代碼,否則返回NULL。
函數(shù)的參數(shù)說明如下:
uDelay:以毫秒指定事件的周期。意味著理論上可以達(dá)到1毫秒的精度.
Uresolution:以毫秒指定延時(shí)的精度,數(shù)值越小定時(shí)器事件分辨率越高。缺省值為1ms。
LpTimeProc:指向一個(gè)回調(diào)函數(shù)。
DwUser:存放用戶提供的回調(diào)數(shù)據(jù)。
FuEvent:指定定時(shí)器事件類型:
TIME_ONESHOT:uDelay毫秒后只產(chǎn)生一次事件
TIME_PERIODIC :每隔uDelay毫秒周期性地產(chǎn)生事件。
具體應(yīng)用時(shí),可以通過調(diào)用timeSetEvent()函數(shù),將需要周期性執(zhí)行的任務(wù)定義在LpTimeProc回調(diào)函數(shù) 中(如:定時(shí)采樣、控制等),從而完成所需處理的事件。需要注意的是,任務(wù)處理的時(shí)間不能大于周期間隔時(shí)間。另外,在定時(shí)器使用完畢后, 應(yīng)及時(shí)調(diào)用timeKillEvent()將之釋放。
2、回調(diào)函數(shù):
void CALLBACK TimeProc(UINT uID,UINT uMsg,DWORD dwUser,DWORD dw1,DWORD dw2);
參數(shù)uID是該多媒體定時(shí)器的標(biāo)識(shí),dwUser與timeSetEvent中的DwUser一致,傳遞回調(diào)函數(shù)中需要使用的參數(shù)。
3、需要注意的問題:
(1)、timeSetEvent在控制臺(tái)程序和窗口程序中都可以運(yùn)行,timeSetEvent執(zhí)行后(若成功)會(huì)啟動(dòng)額外的線程,猜測(cè)這是timeSetEvent可以同時(shí)運(yùn)行在控制臺(tái)和窗口程序中的原因.。
(2)、由于多媒體定時(shí)器是另啟動(dòng)線程處理定時(shí)操作,所以在.回調(diào)函數(shù)中只能訪問本線程的MFC對(duì)象、不能調(diào)用任何系統(tǒng)函數(shù),除了PostMessage, timeGetSystemTime, timeGetTime, timeSetEvent, timeKillEvent, midiOutShortMsg, midiOutLongMsg, OutputDebugString等。
(3)、采用多媒體定時(shí)器時(shí),1s測(cè)試的誤差較大,原因是多媒體定時(shí)器需要啟動(dòng)額外的線程,導(dǎo)致一定的時(shí)間開銷。
二、句柄映射和跨線程訪問
句柄映射:為了防止多個(gè)線程并發(fā)地訪問同一個(gè)MFC對(duì)象,MFC對(duì)象和Windows對(duì)象之間有一個(gè)一一對(duì)應(yīng)的關(guān)系,這種關(guān)系以映射的形式保存在創(chuàng)建線程的當(dāng)前模塊的模塊-線程狀態(tài)信息中。當(dāng)一個(gè)線程使用某個(gè)MFC對(duì)象指針P時(shí),ASSERT_VALID(P)將驗(yàn)證當(dāng)前線程的當(dāng)前模塊是否有Windows句柄和P對(duì)應(yīng),即是否創(chuàng)建了P所指的Windows對(duì)象,驗(yàn)證失敗導(dǎo)致ASSERT斷言中斷程序的執(zhí)行。如果一個(gè)線程要使用其他線程的Windows對(duì)象,則必須傳遞Windows對(duì)象句柄,不能傳遞MFC對(duì)象指針。
但是通常我們需要用定時(shí)器實(shí)現(xiàn)一些定時(shí)更新窗口的命令,更改一些窗口的參數(shù)或者調(diào)用窗口的函數(shù),準(zhǔn)確地說這些都不是對(duì)窗口的操作,是對(duì)于窗口對(duì)應(yīng)并綁定的MFC界面包裝對(duì)象的操作。但是由于句柄映射的機(jī)制,跨線程傳遞MFC界面包裝對(duì)象的指針并在自己的線程中使用是不正確的,通過實(shí)驗(yàn)發(fā)現(xiàn),如果更改該對(duì)象的參數(shù)和自定義函數(shù)結(jié)果是不確定的,很可能產(chǎn)生正確的結(jié)果,但是調(diào)用該MFC類繼承的函數(shù)就會(huì)出現(xiàn)異常。那么如何達(dá)到更新窗口的效果呢,資料顯示有兩種辦法:
1、 通過發(fā)消息的方法轉(zhuǎn)到UI線程去處理,用sendMessage給窗口發(fā)送自定義消息并設(shè)置自己的消息處理函數(shù)來實(shí)現(xiàn)這些功能,窗口收到消息之后調(diào)用與之綁定的MFC界面包裝對(duì)象的消息處理函數(shù)進(jìn)行處理。這種辦法是符合windows機(jī)制并且是線程安全的,但是由于要多發(fā)送至少一條消息,所以犧牲了效率。
2、 傳遞窗口句柄給自定義的線程,并在線程中通過FromHandle()函數(shù)聲稱一個(gè)臨時(shí)界面包裝對(duì)象與窗口句柄綁定,這樣也可以操作該窗口,但是它的派生類功能就消失了,也就是說通過FromHandle()生成的窗口只能是CWnd的實(shí)例,不具有自己定義的那些屬性和操作。而Updatedata()函數(shù)由于是MFC自己提供的一個(gè)對(duì)話框數(shù)據(jù)交換機(jī)制(DDX)的操作,不是通過向窗口句柄發(fā)消息來實(shí)現(xiàn)的而是通過虛函數(shù)機(jī)制。因此調(diào)用的將是CWnd::DoDataExchange不是自己派生類DoDataExchange,所以窗口不會(huì)進(jìn)行正常更新。