C++ 关于MFC多线程编程中的一些注意事项 及自定义消息的处理

在多线程编程中,最简单的方法,无非就是利用 AfxBeginThread  来创建一个工作线程,看一下这个函数的说明:

CWinThread* AFXAPI AfxBeginThread(
AFX_THREADPROC pfnThreadProc,
LPVOID pParam,
int nPriority = THREAD_PRIORITY_NORMAL,
UINT nStackSize = 0,
DWORD dwCreateFlags = 0,
LPSECURITY_ATTRIBUTES lpSecurityAttrs = NULL
);

在这个说明中,除第1和第2两个参数外,余下的参数都有默认值。所以,我们在使用的时候,是必须要指定前两个参数的。

其中 第一个参数是 要运行的函数的名称,光写函数名就可以了,不能加引号。

第二个参数,是指定 运行函数的 参数,这个参数的类型为 LPVOID 。所以要运行的函数的在传递过去后,要转化为LPVOID类型才可以。

而要运行的参数还有一个限制,那就是必须返回一个UINT类型的结果。所以要运行的函数的就有一个基本上固定的格式。

UINT RunProce(LPVOID lpParam)

在这里还需要特别说明一下,这个函数不能是实例函数,也就是函数前面是不能有 类限定符:: 的。如果是静态函数也是可以的。

在这个函数中,我们只能使用一个参数,而参数的类型只能是 LPVOID ,可以用一个结构体来封闭多个参数。

余下的问题,就不是很多了。

关于 多线程,就写到这里吧!

在多线程编程中,一个很重要的问题就是,要将线程的运行过程通知界面线程,做一些显示方面的更新。如下载线程,在适当的时候,可以更新界面,现在下载到什么进度了。等等的情况。但是在工作线程中,是不是直接操作界面线程的控件的。那怎么办呢,只能通过自定义一个消息来解决。

工作流程,就是 在自定义线程中 通过发送一个界面上的 消息,来通知界面做一些更新操作。在这个自定义消息中,有一个细节要解决,那就是自定义消息,必须要指定接收消息的控件句柄。当然你中以使用m_pApp 直接通知主框架来解决,但是这样解决似乎绕了一个很大的圈。其实解决的方法很简单,那就是直接将接收消息的控件的句柄传给自定义线程,就可以了。我们直接在线程中使用此句柄就可以解决了。

我们知道控件的基类都是 CWnd。所以我们传递一个CWnd的指针进去。当然还有一些其它的参数要一块传递进去,那就做一个结构吧

typedef struct{
 CString srcString;
 CString DesString;
 CWnd*    hander;
}Param;

这里我们传递了三个参数 两个字符串一个指针。

我们先造一个自定义线程函数

UINT RunProce(LPVOID lpParam)
{
    Param* par;
    CWnd* hander;
    par = (Param*)lpParam;
    hander = par->hander;
    myCopyDirectory(lpParam);

    CString str;
    str = "复制完成";

    hander->SendMessage(WM_USERMESSAGE,0,(LPARAM)&str);

    return 0;
}

在这个函数中,我们要运行由此函数组成的一个线程的话,就需要传递一个参数lpParam,而这个参数是由 Param 的结构体来指定。实际上是传递了三个参数进去。

Param* par;
par = (Param*)lpParam;

我们会用上在的强制类型转换的方法,就可以还原参数的值。根据这三个参数就  自定主的线程函数就可以运行了。那如何通知界面线程呢。看一下自定义函数里面的这一句

hander->SendMessage(WM_USERMESSAGE,0,(LPARAM)&str);

这一句中 hander 是由结构体转换而来的 接收消息的控件的句柄。然后调用这个控件的 SendMessage 方法,就可以向此控件发消息了。消息的内容由后面的参数来决定

第一个参数 WM_USERMESSAGE 这是一个消息的名称。这个名称实际上是一个数字。我们需要在 .h 文件中 指定一下如下面的格式

#define WM_USERMESSAGE 11130

后面的数字造的大一点,哈哈

第二个与第三个参数,就是这个消息传递具体的值,如果不需要传递值的话,那就直接写0吧

在这里我们想在传递参数的第三个参数上传递一个 字符串,那就是上面的写法了。

这样的话,在线程中发送消息的部分,就全部讲完了。消息发送出去了,怎么接收呢?

这真是一个重要的问题

首先,要将消息做一下映射。消息映射的目的,就是告诉程序,当出现这个消息的时候,使用哪个函数进行处理。这样的话,就首先需要一个消息映射的函数。这个消息映射的函数不是乱写,因为要传递两个参数,所以这个函数需要能够接收这两个参数。处理函数一般这样子写

LRESULT CCopyfileDlg::OnProcName(WPARAM wParam, LPARAM lParam)

他奶奶的,太神奇了。返回值只能是 LRESULT 。这个不用讨论吧,照着抄吧。函数名称后面有参数两个,这是一个实例函数。因为前面有::
两个参数一般也写成这个样子的。

函数内容,就由你的程序的功能决定了。我这里直接抄一段我自己的代码吧

LRESULT CCopyfileDlg::OnProcName(WPARAM wParam, LPARAM lParam)
{
    // TODO: 处理用户自定义消息
    CString* str = (CString*)lParam;
    SetDlgItemText(IDC_STATIC,*str);

    if(*str == "复制完成")
      {
         (CButton*)GetDlgItem(IDC_COPYBUT)->EnableWindow(true);
      }

    return 0;

}

这段程序是根据得到的传递过来的参数,在界面上显示具体的参数内容。

SetDlgItemText(IDC_STATIC,*str);  //在静态文本框中显示消息。

备注:
如果要让按钮变成灰色的,那就使用控件的 EnableWindow 方法。

这个方法,我们说,是专门的消息处理函数,那么它的声明也比较特殊。需要这么写

afx_msg LRESULT OnProcName(WPARAM wParam, LPARAM lParam);

将上面的内容放在 h文件的合理位置就可以了。

现在消息处理函数也有了。但是怎么将映射呢?

其实在 CPP文件中,有一个由 BEGIN_MESSAGE_MAP(CCopyfileDlg, CDialog) 和END_MESSAGE_MAP() 包括的区域。这个区域就是用来定义消息映射的。

将这么一句话放在他们中间,就OK了

ON_MESSAGE(WM_USERMESSAGE,OnProcName)

这么一句话,就将 WM_USERMESSAGE 与 OnProcName 与消息处理函数结合在一起了。是不是超级简单呀!

这样我们的界面线程中的消息处理部分也主做好了。

当消息发送过来后,就会通过消息映射放在对应的函数中加以处理。

时间: 2024-08-07 18:58:42

C++ 关于MFC多线程编程中的一些注意事项 及自定义消息的处理的相关文章

java多线程编程中实现Runnable接口方法相对于继承Thread方法的优势

 java多线程创建方法http://blog.csdn.net/cjc211322/article/details/24999163  java创建多线程方法之间的区别http://blog.csdn.net/cjc211322/article/details/25000449 java多线程编程中实现Runnable接口方法相对于继承Thread方法的优势

Java多线程编程中Future模式的详解<转>

Java多线程编程中,常用的多线程设计模式包括:Future模式.Master-Worker模式.Guarded Suspeionsion模式.不变模式和生产者-消费者模式等.这篇文章主要讲述Future模式,关于其他多线程设计模式的地址如下:关于其他多线程设计模式的地址如下:关于Master-Worker模式的详解: Java多线程编程中Master-Worker模式的详解关于Guarded Suspeionsion模式的详解: Java多线程编程中Guarded Suspeionsion模式

Java多线程编程中的lock使用源码详解

将做工程过程重要的代码段做个记录,如下的代码内容是关于Java多线程编程中的lock使用详解的代码,应该是对码农有帮助. import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Future; import java.util.concurrent.locks.Lock; import java.util.concurrent.l

多线程编程中的EventWaitHandler

首先如果读者是.Net多线程编程的老手,就不用看这篇文章了,这篇文章主要是阐述EventWaitHandler的一些基本原理和用法. 在.NET的System.Threading命名空间中有一个名叫WaitHandler的类,这是一个抽象类(abstract),我们无法手动去创建它,但是WaitHandler有三个子类,这三个子类分别是:System.Threading.EventWaitHandle,System.Threading.Mutex,System.Threading.Semapho

多线程编程中条件变量和的spurious wakeup 虚假唤醒

1. 概述 条件变量(condition variable)是利用共享的变量进行线程之间同步的一种机制.典型的场景包括生产者-消费者模型,线程池实现等. 对条件变量的使用包括两个动作: 1) 线程等待某个条件, 条件为真则继续执行,条件为假则将自己挂起(避免busy wait,节省CPU资源): 2) 线程执行某些处理之后,条件成立:则通知等待该条件的线程继续执行. 3) 为了防止race-condition,条件变量总是和互斥锁变量mutex结合在一起使用. 一般的编程模式: C++代码  

MFC 网络编程中::connect返回-1问题

在MFC编写网络时遇到了::connect总是返回-1,但是与服务器可以进行接收和发送消息的操作. 原因是在进行连接的时候我没有进行初始化:::WSAStartup(w, &data);//动态链接库初始化 在MFC中进行重新连接是也要进行初始化且要先关闭套接字: ::closesocket(s); ::WSACleanup();

【转】Java多线程编程中易混淆的3个关键字( volatile、ThreadLocal、synchronized)总结

概述 最近在看<ThinKing In Java>,看到多线程章节时觉得有一些概念比较容易混淆有必要总结一下,虽然都不是新的东西,不过还是蛮重要,很基本的,在开发或阅读源码中经常会遇到,在这里就简单的做个总结. 1.volatile volatile主要是用来在多线程中同步变量. 在一般情况下,为了提升性能,每个线程在运行时都会将主内存中的变量保存一份在自己的内存中作为变量副本,但是这样就很容易出现多个线程中保存的副本变量不一致,或与主内存的中的变量值不一致的情况.而当一个变量被volatil

c/c++多线程编程中最好不要加volatile

来自https://www.zhihu.com/question/31459750 答主解释说:不能指望volatile能解决多线程竞争问题,除非所用的环境系统不可靠才会为了保险加上volatile, 或者从极限效率考虑来实现很底层的接口,这要求编写者对逻辑走向很清楚,不然会出错. c++11标准明确指出解决多线程的数据竞争问题应该使用原子操作或互斥锁. c和c++中的volatile并不是用来解决多线竞争问题的,而是用来修饰一些程序不可控因素导致变化的变量,比如访问底层硬件设备的变量,来提醒编

mfc多线程编程的几点经验

转自:http://blog.csdn.net/bizhu12/article/details/6832193 1) 不要在线程函数体内操作MFC控件,不要再线程里面调用UpdateData函数更新用户界面,而应该尽量采用发送消息的方式,在主线程的消息响应函数中操作控件: 2)不建议采用SendMessage往主线程发送消息,因为它是同步的,阻塞的,可以考虑采用PostMessage代替: 3)线程退出时,尽量不要使用TerminateThread函数,而尽可能的让线程自己退出: 4) 当线程退