C++ 关于MFC多线程编程的注意事项
在多线程编程中,最简单的方法,无非就是利用AfxBeginThread 来创建一个工作线程,看一下这个函数的说明:
CWinThread*AFXAPIAfxBeginThread( AFX_THREADPROCpfnThreadProc, LPVOIDpParam, intnPriority=THREAD_PRIORITY_NORMAL, UINTnStackSize=0, DWORDdwCreateFlags=0, LPSECURITY_ATTRIBUTESlpSecurityAttrs=NULL );
在这个说明中,除第1和第2两个参数外,余下的参数都有默认值。所以,我们在使用的时候,是必须要指定前两个参数的。
其中第一个参数是要运行的函数的名称,光写函数名就可以了,不能加引号。
第二个参数,是指定运行函数的参数,这个参数的类型为LPVOID。所以要运行的函数的在传递过去后,要转化为LPVOID类型才可以。
而要运行的参数还有一个限制,那就是必须返回一个UINT类型的结果。所以要运行的函数的就有一个基本上固定的格式。
UINTRunProce(LPVOIDlpParam)
在这里还需要特别说明一下,这个函数不能是实例函数,也就是函数前面是不能有类限定符::的。如果是静态函数也是可以的。
在这个函数中,我们只能使用一个参数,而参数的类型只能是LPVOID,可以用一个结构体来封闭多个参数。
余下的问题,就不是很多了。
关于多线程,就写到这里吧!
在多线程编程中,一个很重要的问题就是,要将线程的运行过程通知界面线程,做一些显示方面的更新。如下载线程,在适当的时候,可以更新界面,现在下载到什么进度了。等等的情况。但是在工作线程中,是不是直接操作界面线程的控件的。那怎么办呢,只能通过自定义一个消息来解决。
工作流程,就是在自定义线程中通过发送一个界面上的消息,来通知界面做一些更新操作。在这个自定义消息中,有一个细节要解决,那就是自定义消息,必须要指定接收消息的控件句柄。当然你中以使用m_pApp直接通知主框架来解决,但是这样解决似乎绕了一个很大的圈。其实解决的方法很简单,那就是直接将接收消息的控件的句柄传给自定义线程,就可以了。我们直接在线程中使用此句柄就可以解决了。
我们知道控件的基类都是CWnd。所以我们传递一个CWnd的指针进去。当然还有一些其它的参数要一块传递进去,那就做一个结构吧
typedefstruct{ CStringsrcString; CStringDesString; CWnd* hander; }Param;
这里我们传递了三个参数两个字符串一个指针。
我们先造一个自定义线程函数
UINTRunProce(LPVOIDlpParam) { Param*par; CWnd*hander; par=(Param*)lpParam; hander=par->hander; myCopyDirectory(lpParam); CStringstr; str="复制完成"; hander->SendMessage(WM_USERMESSAGE,0,(LPARAM)&str); return0; }
在这个函数中,我们要运行由此函数组成的一个线程的话,就需要传递一个参数lpParam,而这个参数是由Param的结构体来指定。实际上是传递了三个参数进去。
Param*par; par=(Param*)lpParam;
我们会用上在的强制类型转换的方法,就可以还原参数的值。根据这三个参数就 自定主的线程函数就可以运行了。那如何通知界面线程呢。看一下自定义函数里面的这一句
hander->SendMessage(WM_USERMESSAGE,0,(LPARAM)&str);
这一句中hander是由结构体转换而来的接收消息的控件的句柄。然后调用这个控件的SendMessage方法,就可以向此控件发消息了。消息的内容由后面的参数来决定
第一个参数WM_USERMESSAGE这是一个消息的名称。这个名称实际上是一个数字。我们需要在.h文件中指定一下如下面的格式
#defineWM_USERMESSAGE11130
后面的数字造的大一点,哈哈
第二个与第三个参数,就是这个消息传递具体的值,如果不需要传递值的话,那就直接写0吧
在这里我们想在传递参数的第三个参数上传递一个字符串,那就是上面的写法了。
这样的话,在线程中发送消息的部分,就全部讲完了。消息发送出去了,怎么接收呢?
这真是一个重要的问题
首先,要将消息做一下映射。消息映射的目的,就是告诉程序,当出现这个消息的时候,使用哪个函数进行处理。这样的话,就首先需要一个消息映射的函数。这个消息映射的函数不是乱写,因为要传递两个参数,所以这个函数需要能够接收这两个参数。处理函数一般这样子写
LRESULTCCopyfileDlg::OnProcName(WPARAMwParam,LPARAMlParam)
他奶奶的,太神奇了。返回值只能是LRESULT。这个不用讨论吧,照着抄吧。函数名称后面有参数两个,这是一个实例函数。因为前面有::
两个参数一般也写成这个样子的。
函数内容,就由你的程序的功能决定了。我这里直接抄一段我自己的代码吧
LRESULTCCopyfileDlg::OnProcName(WPARAMwParam,LPARAMlParam) { //TODO:处理用户自定义消息 CString*str=(CString*)lParam; SetDlgItemText(IDC_STATIC,*str); if(*str=="复制完成") { (CButton*)GetDlgItem(IDC_COPYBUT)->EnableWindow(true); } return0; }
这段程序是根据得到的传递过来的参数,在界面上显示具体的参数内容。
SetDlgItemText(IDC_STATIC,*str); //在静态文本框中显示消息。
备注:
如果要让按钮变成灰色的,那就使用控件的EnableWindow方法。
这个方法,我们说,是专门的消息处理函数,那么它的声明也比较特殊。需要这么写
afx_msgLRESULTOnProcName(WPARAMwParam,LPARAMlParam);
将上面的内容放在h文件的合理位置就可以了。
现在消息处理函数也有了。但是怎么将映射呢?
其实在CPP文件中,有一个由BEGIN_MESSAGE_MAP(CCopyfileDlg,CDialog)和END_MESSAGE_MAP()包括的区域。这个区域就是用来定义消息映射的。
将这么一句话放在他们中间,就OK了
ON_MESSAGE(WM_USERMESSAGE,OnProcName)
这么一句话,就将WM_USERMESSAGE与OnProcName与消息处理函数结合在一起了。是不是超级简单呀!
这样我们的界面线程中的消息处理部分也主做好了。
当消息发送过来后,就会通过消息映射放在对应的函数中加以处理。
以上所述就是本文的全部内容了,希望大家能够喜欢。