一直以來,對在工作線程中更新UI這個問題沒有一個全面的認識,看到下面的文章,總算解決了心中長久以來的疑惑;豁然開朗。
個人比較喜歡第一種方法。
http://www.itepub.net/html/kaifajingcui/C___VC/2006/0515/38137.html
最近寫了個代碼,在UI線程中創建了一個窗口,然后在工作線程中修改了這個窗口中的一些數據,然后想用UpdateData(FALSE)來更新窗口的內容,結果在Debug版本下面就出現了Assert報錯,說出錯地方是wincore.cpp的888行和889行,就是這兩句
ASSERT((p = pMap->LookupPermanent(m_hWnd)) != NULL ||
(p = pMap->LookupTemporary(m_hWnd)) != NULL);
ASSERT((CWnd*)p == this); // must be us
我用MFC也沒有多久,也不太熟悉,翻了翻資料,在http://support.microsoft.com/default.aspx?scid=kb;en-us;147578找到一篇文章,就是說MFC窗口跨線程的問題的,大概意思就是MFC的窗口是線程相關的,每個窗口的HandleMap是儲存在線程相關的堆棧里面的 (thread-local-storage (TLS) ),那這樣我就理解了為什么上面兩句ASSERT會出錯了,線程環境都切換了當然線程堆棧的數據也就不一樣了.
這篇文章提供了兩種修改方案:
一種是用FromHandle來獲得一個CWnd*,然后再調用UpdateData,這個方案我沒有實驗成功,結果是錯雖然不報了,但是界面也沒有被更新.
另外一種是通過發消息的方法轉到UI線程去處理.可以在窗口映射一個消息,比如ON_MESSAGE(WM_UPDATEDATA, OnUpdateData),然后用SendMessage(WM_UPDATEDATA, FALSE)傳消息給窗口,窗口的消息處理肯定是在UI線程里面,這時候可以用
LRESULT CProtectPage::OnUpdateData(WPARAM wParam, LPARAM lParam)
{
UpdateData(wParam);
return 0;
}
來更新界面,實驗是成功的,ASSERT就被消除了.
還是有點疑惑,就是剛開始直接在工作線程中調用UpdateData(FALSE)的時候,雖然有ASSERT報錯,但是結果還是正確的,似乎沒有什么影響,不知道這個ASSERT到底意味著什么?