C++ 關(guān)于MFC多線程編程的注意事項(xiàng)
在多線程編程中,最簡(jiǎn)單的方法,無非就是利用 AfxBeginThread 來創(chuàng)建一個(gè)工作線程,看一下這個(gè)函數(shù)的說明:
CWinThread* AFXAPI AfxBeginThread(
AFX_THREADPROC pfnThreadProc,
LPVOID pParam,
int nPriority = THREAD_PRIORITY_NORMAL,
UINT nStackSize = 0,
DWORD dwCreateFlags = 0,
LPSECURITY_ATTRIBUTES lpSecurityAttrs = NULL
);
在這個(gè)說明中,除第1和第2兩個(gè)參數(shù)外,余下的參數(shù)都有默認(rèn)值。所以,我們?cè)谑褂玫臅r(shí)候,是必須要指定前兩個(gè)參數(shù)的。
其中 第一個(gè)參數(shù)是 要運(yùn)行的函數(shù)的名稱,光寫函數(shù)名就可以了,不能加引號(hào)。
第二個(gè)參數(shù),是指定 運(yùn)行函數(shù)的 參數(shù),這個(gè)參數(shù)的類型為 LPVOID 。所以要運(yùn)行的函數(shù)的在傳遞過去后,要轉(zhuǎn)化為L(zhǎng)PVOID類型才可以。
而要運(yùn)行的參數(shù)還有一個(gè)限制,那就是必須返回一個(gè)UINT類型的結(jié)果。所以要運(yùn)行的函數(shù)的就有一個(gè)基本上固定的格式。
UINT RunProce(LPVOID lpParam)
在這里還需要特別說明一下,這個(gè)函數(shù)不能是實(shí)例函數(shù),也就是函數(shù)前面是不能有 類限定符:: 的。如果是靜態(tài)函數(shù)也是可以的。
在這個(gè)函數(shù)中,我們只能使用一個(gè)參數(shù),而參數(shù)的類型只能是 LPVOID ,可以用一個(gè)結(jié)構(gòu)體來封閉多個(gè)參數(shù)。
余下的問題,就不是很多了。
關(guān)于 多線程,就寫到這里吧!
在多線程編程中,一個(gè)很重要的問題就是,要將線程的運(yùn)行過程通知界面線程,做一些顯示方面的更新。如下載線程,在適當(dāng)?shù)臅r(shí)候,可以更新界面,現(xiàn)在下載到什么進(jìn)度了。等等的情況。但是在工作線程中,是不是直接操作界面線程的控件的。那怎么辦呢,只能通過自定義一個(gè)消息來解決。
工作流程,就是 在自定義線程中 通過發(fā)送一個(gè)界面上的 消息,來通知界面做一些更新操作。在這個(gè)自定義消息中,有一個(gè)細(xì)節(jié)要解決,那就是自定義消息,必須要指定接收消息的控件句柄。當(dāng)然你中以使用m_pApp 直接通知主框架來解決,但是這樣解決似乎繞了一個(gè)很大的圈。其實(shí)解決的方法很簡(jiǎn)單,那就是直接將接收消息的控件的句柄傳給自定義線程,就可以了。我們直接在線程中使用此句柄就可以解決了。
我們知道控件的基類都是 CWnd。所以我們傳遞一個(gè)CWnd的指針進(jìn)去。當(dāng)然還有一些其它的參數(shù)要一塊傳遞進(jìn)去,那就做一個(gè)結(jié)構(gòu)吧
typedef struct{
CString srcString;
CString DesString;
CWnd* hander;
}Param;
這里我們傳遞了三個(gè)參數(shù) 兩個(gè)字符串一個(gè)指針。
我們先造一個(gè)自定義線程函數(shù)
UINT RunProce(LPVOID lpParam)
{
Param* par;
CWnd* hander;
par = (Param*)lpParam;
hander = par->hander;
myCopyDirectory(lpParam);
CString str;
str = "復(fù)制完成";
hander->SendMessage(WM_USERMESSAGE,0,(LPARAM)&str);
return 0;
}
在這個(gè)函數(shù)中,我們要運(yùn)行由此函數(shù)組成的一個(gè)線程的話,就需要傳遞一個(gè)參數(shù)lpParam,而這個(gè)參數(shù)是由 Param 的結(jié)構(gòu)體來指定。實(shí)際上是傳遞了三個(gè)參數(shù)進(jìn)去。
Param* par;
par = (Param*)lpParam;
我們會(huì)用上在的強(qiáng)制類型轉(zhuǎn)換的方法,就可以還原參數(shù)的值。根據(jù)這三個(gè)參數(shù)就 自定主的線程函數(shù)就可以運(yùn)行了。那如何通知界面線程呢。看一下自定義函數(shù)里面的這一句
hander->SendMessage(WM_USERMESSAGE,0,(LPARAM)&str);
這一句中 hander 是由結(jié)構(gòu)體轉(zhuǎn)換而來的 接收消息的控件的句柄。然后調(diào)用這個(gè)控件的 SendMessage 方法,就可以向此控件發(fā)消息了。消息的內(nèi)容由后面的參數(shù)來決定
第一個(gè)參數(shù) WM_USERMESSAGE 這是一個(gè)消息的名稱。這個(gè)名稱實(shí)際上是一個(gè)數(shù)字。我們需要在 .h 文件中 指定一下如下面的格式
#define WM_USERMESSAGE 11130
后面的數(shù)字造的大一點(diǎn),哈哈
第二個(gè)與第三個(gè)參數(shù),就是這個(gè)消息傳遞具體的值,如果不需要傳遞值的話,那就直接寫0吧
在這里我們想在傳遞參數(shù)的第三個(gè)參數(shù)上傳遞一個(gè) 字符串,那就是上面的寫法了。
這樣的話,在線程中發(fā)送消息的部分,就全部講完了。消息發(fā)送出去了,怎么接收呢?
這真是一個(gè)重要的問題
首先,要將消息做一下映射。消息映射的目的,就是告訴程序,當(dāng)出現(xiàn)這個(gè)消息的時(shí)候,使用哪個(gè)函數(shù)進(jìn)行處理。這樣的話,就首先需要一個(gè)消息映射的函數(shù)。這個(gè)消息映射的函數(shù)不是亂寫,因?yàn)橐獋鬟f兩個(gè)參數(shù),所以這個(gè)函數(shù)需要能夠接收這兩個(gè)參數(shù)。處理函數(shù)一般這樣子寫
LRESULT CCopyfileDlg::OnProcName(WPARAM wParam, LPARAM lParam)
他奶奶的,太神奇了。返回值只能是 LRESULT 。這個(gè)不用討論吧,照著抄吧。函數(shù)名稱后面有參數(shù)兩個(gè),這是一個(gè)實(shí)例函數(shù)。因?yàn)榍懊嬗校海?br />
兩個(gè)參數(shù)一般也寫成這個(gè)樣子的。
函數(shù)內(nèi)容,就由你的程序的功能決定了。我這里直接抄一段我自己的代碼吧
LRESULT CCopyfileDlg::OnProcName(WPARAM wParam, LPARAM lParam) { // TODO: 處理用戶自定義消息 CString* str = (CString*)lParam; SetDlgItemText(IDC_STATIC,*str); if(*str == "復(fù)制完成") { (CButton*)GetDlgItem(IDC_COPYBUT)->EnableWindow(true); } return 0; }
這段程序是根據(jù)得到的傳遞過來的參數(shù),在界面上顯示具體的參數(shù)內(nèi)容。
SetDlgItemText(IDC_STATIC,*str); //在靜態(tài)文本框中顯示消息。
備注:
如果要讓按鈕變成灰色的,那就使用控件的 EnableWindow 方法。
這個(gè)方法,我們說,是專門的消息處理函數(shù),那么它的聲明也比較特殊。需要這么寫
afx_msg LRESULT OnProcName(WPARAM wParam, LPARAM lParam);
將上面的內(nèi)容放在 h文件的合理位置就可以了。
現(xiàn)在消息處理函數(shù)也有了。但是怎么將映射呢?
其實(shí)在 CPP文件中,有一個(gè)由 BEGIN_MESSAGE_MAP(CCopyfileDlg, CDialog) 和END_MESSAGE_MAP() 包括的區(qū)域。這個(gè)區(qū)域就是用來定義消息映射的。
將這么一句話放在他們中間,就OK了
ON_MESSAGE(WM_USERMESSAGE,OnProcName)
這么一句話,就將 WM_USERMESSAGE 與 OnProcName 與消息處理函數(shù)結(jié)合在一起了。是不是超級(jí)簡(jiǎn)單呀!
這樣我們的界面線程中的消息處理部分也主做好了。
當(dāng)消息發(fā)送過來后,就會(huì)通過消息映射放在對(duì)應(yīng)的函數(shù)中加以處理。
以上所述就是本文的全部?jī)?nèi)容了,希望大家能夠喜歡。
- C++ 中消息隊(duì)列函數(shù)實(shí)例詳解
- C++開發(fā):為什么多線程讀寫shared_ptr要加鎖的詳細(xì)介紹
- C++多線程編程簡(jiǎn)單實(shí)例
- C/C++ 多線程的學(xué)習(xí)心得總結(jié)
- C++實(shí)現(xiàn)多線程查找文件實(shí)例
- linux下的C\C++多進(jìn)程多線程編程實(shí)例詳解
- C++多線程編程時(shí)的數(shù)據(jù)保護(hù)
- c++多線程之死鎖的發(fā)生的情況解析(包含兩個(gè)歸納,6個(gè)示例)
- C++多線程中的鎖和條件變量使用教程
- C++基于消息隊(duì)列的多線程實(shí)現(xiàn)示例代碼
相關(guān)文章
C語言循環(huán)隊(duì)列與用隊(duì)列實(shí)現(xiàn)棧問題解析
循環(huán)隊(duì)列又叫環(huán)形隊(duì)列,是一種特殊的隊(duì)列。循環(huán)隊(duì)列解決了隊(duì)列出隊(duì)時(shí)需要將所有數(shù)據(jù)前移一位的問題,本篇帶你一起看看循環(huán)隊(duì)列的問題和怎樣用隊(duì)列實(shí)現(xiàn)棧2022-04-04關(guān)于C++中vector的兩個(gè)小tips分享
這篇文章主要給大家介紹了關(guān)于C++中vector的兩個(gè)小tips,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用C++具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-05-05詳解C++的靜態(tài)內(nèi)存分配與動(dòng)態(tài)內(nèi)存分配
內(nèi)存分配 (Memory Allocation) 是指為計(jì)算機(jī)程序或服務(wù)分配物理內(nèi)存空間或虛擬內(nèi)存空間的一個(gè)過程,本文主要介紹了C++的靜態(tài)內(nèi)存分配與動(dòng)態(tài)內(nèi)存分配,感興趣的同學(xué)可以參考閱讀2023-06-06