• 添加自定义消息处理


    1.PreTranslateMessage

    PreTranslateMessage是消息在送给TranslateMessage函数之前被调用的。

    头文件中重载该函数

    virtual BOOL PreTranslateMessage(MSG* pMsg);

    源文件中写实现方法

    BOOL CClientDlg::PreTranslateMessage(MSG* pMsg) 

    {
    if (pMsg->message==1000)
    {
    ReceiveData();
    return TRUE;
    }
    else
    return CDialog::PreTranslateMessage(pMsg);

    }

    定义为1000的消息由WSAAsyncSelect(m_client, m_hWnd, 1000, FD_READ)发出,PreTranslateMessage接收到该消息就会接收数据。


    2.PostMessage

    函数原型:BOOL WINAPI      PostMessage(
                                                                                  _In_opt_  HWND hWnd,                      //接收消息的窗口的句柄,NULL表示发送到当前线程窗口
                                                                                  _In_      UINT Msg,                                //指定被寄送的消息
                                                                                  _In_      WPARAM wParam,                //指定附加的消息特定的信息
                                                                                  _In_      LPARAM lParam                    //指定附加的消息特定的信息
                                                                                );
    该函数将一个消息放入(寄送)到与指定窗口创建的线程相联系消息队列里,不等待线程处理消息就返回,是异步消息模式。消息队列里的消息通过调用GetMessage和PeekMessage取得。
    如果函数调用成功,返回非零,函数调用返回值为零。
    头文件:winuser.h;
    输入库:user32.lib;
    1、由于是用户自己定义的消息,所以首先要定义一个消息宏如下:

    #define WM_MYMESSAGE (WM_USER+1)

    注意:为防止用户定义的宏和系统定义宏冲突,所以系统提供了一个WM_USER,只要是大于WM_USER可供用户使用

    2、定义一个消息响应宏,即将此宏映射的函数,如下:

    ON_MESSAGE(WM_MYMESSAGE,OnMyMssage)
    WM_MYMESSAGE为自定义的宏,OnMyMssage为消息处理函数名

    3、在头文件中声明自己的消息处理函数,如下:

    afx_msg LRESULT OnMyMssage(WPARAM w,LPARAM l);

    4、定义消息处理函数,如下:

    LRESULT CPostMessageDlg::OnMyMssage(WPARAM wParam, LPARAM lParam)
    {
         MessageBox(_T("自定义消息"));
         return 1;
    }

    5、通过PostMessage函数将OnMyMssage添加到消息队列中,如下:

    void CPostMessageDlg::OnButton1() //按钮响应事件
    {
         PostMessage(WM_MYMESSAGE,IDC_BUTTON1);//发送之后立即返回
         //SendMessage(WM_MYMESSAGE);//发送之后等待返回
    }


    3.SendMessage

    函数原型
    LRESULT SendMessage(

                                                     HWND hWnd,            //将接收消息的窗口的句柄

                                                     UINT Msg,                  //指定被发送的消息

                                                     WPARAM wParam,  //指定被发送的消息

                                                     LPARAM IParam)      //指定被发送的消息

    该函数将指定的消息发送到一个或多个窗口。此函数为指定的窗口调用窗口程序,直到窗口程序处理完消息再返回。

    例子

    以下这个例子中是一个MFC的对话框应用程序,名字为MessageTest。它包括一个发送对话框和一个接收对话框,其中发送对话框和接收对话框都是主对话框的子对话框,利用消息传递来从发送对话框向接收对话框发送数据。

    头文件

    #define RECEIVE_TITLE "receive title"

    #define GET_STRUCT WM_USER+1000
    struct SendStruct
    {
          int a;
          int b;
          int c;
          int d;
    };

    RECEIVE_TITLE定义接收对话框的标题,GET_STRUCT是消息ID,WM_USER是用户自定义消息的起始ID。在生成的接受对话框的位置加入SetWindowText(RECEIVE_TITLE)。

    1.      在接受对话框1的类的定义中加入
    afx_msg LRESULT GetStruct(WPARAM wparam,LPARAM lparam);
    2.      在对应的cpp文件中加入
    ON_MESSAGE(GET_STRUCT,ReceiveDlg::GetStruct)
    3.      在函数实现部分加入
    LRESULT ReceiveDlg::GetStruct(WPARAM wparam,LPARAM lparam)
    {
          SendStruct* ss=(SendStruct*)wparam;
          CString str;
          str.Format("%d,%d,%d,%d",ss->a,ss->b,ss->c,ss->d);
          CEdit* edit1=(CEdit*)GetDlgItem(IDC_EDIT1);
          edit1->SetWindowText(str);
          free(ss);
          return 0;
    }
    4.      在发送消息的函数中加入
    SendStruct *ss = (SendStruct *)malloc(sizeof(SendStruct));
    ss->a=1;
    ss->b=2;
    ss->c=3;
    ss->d=4;
    HWND hWnd = ::FindWindowEx(this->GetParent()->m_hWnd, NULL, NULL, RECEIVE_TITLE);
    FromHandle(hWnd)->SendMessage(GET_STRUCT,(WPARAM)(ss),0);

    其中,m_hWnd为接收消息的父窗口的句柄,RECEIVE_TITLE为接收消息窗口的标题,得到的hWnd为接收消息窗口的句柄。

    总结:

    PostMessage函数将一个消息放入与创建这个窗口的消息队列相关的线程中,并立刻返回不等待线程处理消息,SendMessage函数将指定的消息发到窗口。它调用特定窗口的窗口处理函数,并且不会立即返回,直到窗口处理函数处理了这个消息。SendMessage的确是发送消息,然后等待处理完成返回,但发送消息的方法为直接调用消息处理函数(即WndProc函数),按照函数调用规则,肯定会等消息处理函数返回之后,SendMessage才返回。而PostMessage却没有发送消息,PostMessage是将消息放入消息队列中,然后立刻返回,至于消息何时被处理,PostMessage完全不知道,此时只有消息循环知道被PostMessage的消息何时被处理了。
    所以SendMessage只是调用我们的消息处理函数,PostMessage只是将消息放到消息队列中。更直接的说SendMessage的消息是不进队列的,而PostMessage的需要排队。

    我们可以PreTranslateMessage来对消息预处理,该用的用,不该用的不用。用SendMessage发送的消息是不能用PreTranslateMessage来预处理的,因为SendMessage的消息是不进队列的。而Post的就可以。

    版权声明:

  • 相关阅读:
    C编程规范
    c# 闭包 小例
    计算前后2行的时间差
    判断是不是奇数
    条件表达式工具类
    代码重构-5 取消类的私有变量(实例变量)
    代码重构-4 通用方法 用 static
    代码重构-3 用Tuple代替 out与ref
    代码重构-2 简单不变的 if else 用字典代替
    代码重构-1 对参数中有 bool值的拆分
  • 原文地址:https://www.cnblogs.com/walccott/p/4957110.html
Copyright © 2020-2023  润新知