python sendmessage_用户自定义消息SendMessage的使用

python sendmessage_用户自定义消息SendMessage的使用大家好 博主在这里跟大家分享一下学习 MFC 的经验 关于 MFC 的每篇帖子我都会奉上原文档下载地址 百度文库 和源代码的下载地址 CSDN 都是免金币或积分的 目的就是更加有利于与大家交流 共同进步 因为是初学 所以肯定有疏漏之处 请大家指正 SendMessage 可以理解为 SendMessage 函数发送消息 等待消息处理完成后 SendMessage 才返回 稍微深入一点 是等待窗口处理函数返回后

大家好,博主在这里跟大家分享一下学习MFC的经验。关于MFC的每篇帖子我都会奉上原文档下载地址(百度文库)和源代码的下载地址(CSDN),都是免金币或积分的!目的就是更加有利于与大家交流,共同进步!因为是初学,所以肯定有疏漏之处,请大家指正

SendMessage可以理解为,SendMessage函数发送消息,等待消息处理完成后,SendMessage才返回。稍微深入一点,是等待窗口处理函数返回后,SendMessage就返回了。

PostMessage可以理解为,PostMessage函数发送消息,不等待消息处理完成,立刻返回。稍微深入一点,PostMessage只管发送消息,消息有没有被送到则并不关心,只要发送了消息,便立刻返回

这里主要讲一下mfc中SendMessage的使用方法。传递消息主要分4步:

1.  在类的定义中声明消息函数:afx_msg void AAA();

2.  在相应的cpp文件中的MESSAGE_MAP区域内添加ON_MESSAGE(MESSAGE_ID,AAA),其中参数1为要传递消息的ID,参数2为刚刚声明的函数名称,不用带括号。

3.  实现消息函数:在cpp文件中添加

LRESULT 类名::AAA(WPARAM

wparam,LPARAM lparam) {

执行内容

return

0;

}

4.  发送消息:在需要发送消息的地方添加下列语句:

HWND hWnd = ::FindWindowEx( m_hWnd, NULL, NULL, WINDOW_TEXT ) ;

FromHandle(hWnd)->SendMessage(MESSAGE_ID,a,b);

其中,m_hWnd为接收消息的父窗口的句柄,WINDOW_TEXT为接收消息窗口的标题,得到的hWnd为接收消息窗口的句柄。调用该窗口的SendMessage函数,MESSAGE_ID为刚刚设定的消息ID,a和b是要传递的参数。

注:在这4个步骤中,前三个我在做的时候基本没什么障碍。问题主要出现在第4步。开始找到网上的例子给的都是FindWindow函数,怎么用都不好使。后来看到有人说FindWindow是找操作系统下打开的窗口的句柄,找窗口中子窗口要用FindWindowEx函数。我也尝试过用对话框的ID找到相应的句柄,像GetDlgItem(ID)函数一样,未果。我使用的对话框都是没有标题栏的,所以也就没有窗口的标题,当然这并不会影响我设置标题。只要在生成该窗口的区域内添加SetWindowText(“窗口标题”)就可以了。也就是说窗口标题可以设置,但是不会显示。最后一点在SendMessage()函数中,MFC默认传递的参数是WPARAM和LPARAM型(一个是UINT型,一个LONG型),如果要传递浮点类型,或者其它不是整数的类型,就可以用指针的形式传递(如果发送方只是申请一个变量并以地址的形式传递,然后接收方以指针的形式接收,如果在执行完SendMessage之后原函数体立即结束了,我不知道在接收函数体接收和使用该变量的之间的一瞬间,该内存区域会不会被占用,我觉得还是有这种可能的。所以我觉得还是在原函数体先申请一块内存,然后在接收函数使用完之后再释放该内存比较合理吧)。

在以下这个例子中是一个MFC的对话框应用程序,名字为MessageTest。它包括左边的一个发送对话框,和右边的两个接收对话框,其中发送对话框和接收对话框1分别是主对话框的子对话框,在接受对话框中有一个Tab Control,在Tab Control中有个接受对话框2。这么做的目的主要是为了理解如何找句柄的,为此我把几个对话框设置成深陷下去的便于观察。

准备工作:

1.  手动添加1个设置全局变量的头文件GlobalSetting.h,这样做的目的是让所有的地方都能知道对话框的标题和自定义的结构。

2.  在GlobalSetting.h中加入下列语句:

#define RECEIVE1_TITLE “receive1 title”

#define RECEIVE2_TITLE “receive2 title”

#define GET_INT WM_USER+1000

#define GET_DOUBLE WM_USER+1001

#define GET_STRING WM_USER+1002

#define GET_STRUCT WM_USER+1003

struct SendStruct

{

int  a;

int  b;

int  c;

int  d;

};

RECEIVE1_TITLE和RECEIVE2_TITLE定义两个接收对话框的标题,后边的4个GET_是4个消息ID,WM_USER是用户自定义消息的起始ID。

3.  在各个需要使用这些内容的位置添加#include “GlobalSetting.h”。

4.  在生成两个接受对话框的位置加入SetWindowText(RECEIVE1_TITLE)和SetWindowText(RECEIVE2_TITLE)。

4aff4a202edf658e1c856420f075859e.png

例1:从发送对话框发送整数到接受对话框1中。

1.  在接受对话框1的类的定义中加入

afx_msg LRESULT

GetInt(WPARAM wparam,LPARAM lparam);

2.  在对应的cpp文件中加入

ON_MESSAGE(GET_INT,Receive1Dlg::GetInt)

3.  在函数实现部分加入

LRESULT Receive1Dlg::GetInt(WPARAM wparam,LPARAM

lparam) {

CString  str; str.Format(“%d %d”,wparam,lparam);

CEdit*  edit1=(CEdit*)GetDlgItem(IDC_EDIT1); edit1->SetWindowText(str);

return  0;

}

4. 在发送消息的函数中加入

int a=1; int b=2;

HWND hWnd = ::FindWindowEx(

this->GetParent()->m_hWnd, NULL, NULL, RECEIVE1_TITLE);

FromHandle(hWnd)->SendMessage(GET_INT,a,b);

注:在这个部分中只是简单地传递两个整数。第3步是将得到的两个整数显示到编辑框中,wparam和lparam如果是整数可以直接使用。第4步中this->GetParent()->m_hWnd是找到了接收对话框1父窗口的句柄,将a和b的值发过去。

例2:从发送对话框发送小数到接受对话框1中。

1.  在接受对话框1的类的定义中加入

afx_msg LRESULT

GetDouble(WPARAM wparam,LPARAM lparam);

2.  在对应的cpp文件中加入

ON_MESSAGE(GET_DOUBLE,Receive1Dlg::GetDouble)

3.  在函数实现部分加入

LRESULT

Receive1Dlg::GetDouble(WPARAM wparam, LPARAM lparam)

{CString  str;

double*  a=(double*)wparam; str.Format(“%lf”,*a);

CEdit*

edit1=(CEdit*)GetDlgItem(IDC_EDIT1); edit1->SetWindowText(str);

delete  a; return  0;

}

4.  在发送消息的函数中加入

double *a=new  double;

*a=1.111; HWND hWnd =

::FindWindowEx( this->GetParent()->m_hWnd, NULL, NULL,

RECEIVE1_TITLE);

FromHandle(hWnd)->SendMessage(GET_DOUBLE,(WPARAM)(a),0);

注:在这个部分中是传递1个小数。第3步是将得到的小数显示到编辑框中。第4步中先为double类型的变量申请一块内存并为其负值,然后以指针的形式发送消息过去。在第3步中首先获得double型指针,最后将内存释放。

例3:从发送对话框发送字符串到接受对话框2中。

1.  在接受对话框2的类的定义中加入

afx_msg LRESULT

GetString(WPARAM wparam,LPARAM lparam);

2.  在对应的cpp文件中加入

ON_MESSAGE(GET_STRING,Receive2Dlg::GetString)

3.  在函数实现部分加入

LRESULT Receive2Dlg::GetString(WPARAM wparam,LPARAM

lparam)

{ CString  *str=(CString*)wparam;

CEdit*  edit1=(CEdit*)GetDlgItem(IDC_EDIT1); edit1->SetWindowText(*str);

delete  str;

return  0;

}

4.  在发送消息的函数中加入

CString  *p_str=new CString(“Hello”);

HWND hWnd =  ::FindWindowEx(this->GetParent()->m_hWnd, NULL, NULL,

RECEIVE1_TITLE);

hWnd=::FindWindowEx(FromHandle(hWnd)->GetDlgItem(IDC_TAB1)->m_hWnd,

NULL, NULL, RECEIVE2_TITLE);

FromHandle(hWnd)->SendMessage(GET_STRING,(WPARAM)(p_str),0);

注:在这个部分中是传递1个字符串。第3步是将得到的字符串显示到编辑框中。第4步中先为CString类型的变量申请一块内存并为其赋值,然后以指针的形式发送消息过去。这个我在获得对话框句柄的时候遇到了一些问题,首先一定要弄清楚各个窗口之间的父子关系,并知道对话框和控件获得句柄的方法是不同的。在这个例子中,接收对话框2的父窗口是那个Tab Control,Tab Control的父窗口是接收对话框1,接收对话框1和发送对话框拥有共同的父窗口。调理清晰了就好写了。HWND hWnd = ::FindWindowEx(this->GetParent()->m_hWnd,

NULL, NULL, RECEIVE1_TITLE);前边已经说过了,是获得了接收对话框1的句柄,下边那条语句中的FromHandle(hWnd)->GetDlgItem(IDC_TAB1)->m_hWnd是获得Tab Control的句柄,hWnd= ::FindWindowEx (FromHandle(hWnd)->

GetDlgItem(IDC_TAB1)-> m_hWnd, NULL, NULL, RECEIVE2_TITLE);就是获得接收对话框2的句柄了。

例4:从发送对话框发送结构到接受对话框2中。

1.  在接受对话框1的类的定义中加入

afx_msg LRESULT GetStruct(WPARAM wparam,LPARAM

lparam);

2. 在对应的cpp文件中加入 ON_MESSAGE(GET_STRUCT,Receive2Dlg::GetStruct)

3.  在函数实现部分加入

LRESULT Receive2Dlg::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,

RECEIVE1_TITLE);

hWnd = ::FindWindowEx( FromHandle(hWnd)->

GetDlgItem(IDC_TAB1)-> m_hWnd, NULL, NULL ,RECEIVE2_TITLE);

FromHandle(hWnd)->SendMessage(GET_STRUCT,(WPARAM)(ss),0);

这个就不多做解释了。

错误的情况:下面说一下我在开始使用消息时使用错误的情况(我认为这是错的)。拿例2的第4步来说:

double

a=1.1111;

HWND hWnd =

::FindWindowEx( this->GetParent()->m_hWnd, NULL, NULL,

RECEIVE1_TITLE);

FromHandle(hWnd)->SendMessage(GET_DOUBLE,(WPARAM)(&a),0);

第3步中去掉delete a;

我认为这是错的,为什么呢?很多时候在发送完成消息之后,函数体就结束了,a的内存就会被收回了,而消息接收的函数可能在很久的时间内都使用这个变量,说不定什么时候a的内存就被重新占用了。如果a是一个很庞大的对象,这种现象可能会更明显。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/222565.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月17日 下午3:30
下一篇 2026年3月17日 下午3:30


相关推荐

  • SpringBoot集成Redis,并自定义对象序列化

    SpringBoot集成Redis,并自定义对象序列化SpringBoot项目使用redis非常简单,pom里面引入redis的场景启动器,在启动类上加@EnableCaching注解,项目启动会自动匹配上redis,这样项目中就可以愉快地使用了,使用方法:要么使用@Cacheable一类的注解自动缓存,要么使用RedisTemplate手动缓存。(前提是你的本机或者是远程主机要先搭好redis环境)虽然SpringBoot好用,但这里也有好多…

    2026年2月14日
    5
  • 2022年0425 ICEM网格划分学习笔记[通俗易懂]

    2022年0425 ICEM网格划分学习笔记[通俗易懂]ICEM学习笔记

    2022年5月25日
    53
  • java课程设计-多人聊天工具(socket+多线程)

    大一下学期的java期末课程设计,分享一下文章目录课设要求相关知识点类图项目框架核心代码1.服务器端Server.java课设要求多人聊天工具服务器要求1:能够看到所有在线用户(25%)服务器要求2:能够强制用户下线(25%)客户端要求1:能够看到所有在线用户(25%)客户端要求2:能够向某个用户发送消息(25%)相关知识点1.服务端能够看到所有在线用户服务端继承了JFrame,实现可视化,通过socket实现服务端与客户端的连接,服务端每接收一个连接,把传进来的用户名和对应的s.

    2022年4月7日
    63
  • 中国电信修改光猫路由模式为桥接模式怎么改_打10000改电信光猫桥接模式

    中国电信修改光猫路由模式为桥接模式怎么改_打10000改电信光猫桥接模式首先,需要搞到超级管理员的账号和密码,可以上网根据光猫型号查找,也可以直接跟安宽带的工作人员要.第一步:准备超级管理员账号和密码.可以自行根据光猫型号搜索,也可以直接跟安宽带的工作人员要.第二步:使用超级管理员账密登录网关管理页面(光猫)一般为192.168.1.1(前提你的路由模式为路由模式,而不是桥接模式)第三步:修改光猫参数1:连接名称修改改为3_INTERNET_R_VID_412:连接模式修改为桥接3:保存第四步:使用路由器拨号宽带账号和密码一般都会贴在光猫上,如果丢失了

    2022年10月8日
    14
  • macOS 10.12 不允许未知来源开发者的应用

    macOS 10.12 不允许未知来源开发者的应用在终端输入 sudospctlmas disable 注意 master 前面是两个 因为有的时候会出现两个 合成一个 所以说明一下

    2026年3月7日
    3
  • 青龙面板使用_京东薅羊毛 青龙面板

    青龙面板使用_京东薅羊毛 青龙面板轻量服务器—-青龙面板开启京东白嫖1.拉取镜像,部署dockerpullwhyour/qinglong:latestdockerrun-dit\-v$PWD/ql/config:/ql/config\-v$PWD/ql/log:/ql/log\-v$PWD/ql/db:/ql/db\-p5700:5700\–nameqinglong\–hostnameqinglong\–restartalways

    2026年1月17日
    4

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号