消息宏

1、命令消息(WM_COMMAND)
所有派生自 CCmdTarget 的类都有资格接受WM_COMMAND。

2、Window消息(WM_xxx)
所有派生自 CWnd 的类都有资格接受 WM_xxx。

3、控件消息(WM_NOTIFY)
控件向其父窗口通知消息。

消息处理
1、WM_xxx 消息处理
窗口类(自身)处理→基类处理→CWnd∷DefWindowProc()处理;
其所对应的宏一般为在消息 WM_ 前面加上 ON_。

2、命令消息处理
命令消息来自命令用户接口对象(菜单、加速键或工具栏按钮)发出的WM_COMMAND消息;
㈠、WM_COMMAND消息
其所包含的类型和对应的宏如下:
①、ON_COMMAND(ID,pfn)
标准的命令消息;

②、ON_COMMAND_EX(ID,pfn)
多个对象对同一个命令 ID 的处理;
其函数的原型如下:
afx_msg BOOL pfn(UINT nID)
说明:
当返回 TRUE 时表示已经处理,不用在消息处理链中继续处理该命令;为 FALSE 时表示继续在消息处理链中处理该命令。
注意:
其一:在多对象处理中一定要使用该宏;
其二:pfn(UINT nID)(消息处理函数)返回值将其类型void改成BOOL,而且必须为FALSE;
其三:多个对象的处理是由高层向低层的过程:即视图类→主框架窗口类→应用程序类;

③、ON_COMMAND_RANGE(nID,nLastID,pfn)
多个命令 ID 提供相同的处理;
注意:
其一:确保nID、nLastID的值在 Resource.h 中是连续的。
其二:一般在函数 pfn(UINT nID) 中加入参数,用来确定那一个按钮点击。

㈡、CN_UPDATE_COMMAND_UI消息
当菜单项、工具栏按钮等[命令用户接口对象]要更新其状态时所对应的消息,它所包含的类型和对应的宏如下:
①、ON_UPDATE_COMMAND_UI(ID,pfn)
其中函数的原型如下:
afx_msg void pfn(CCmdUI* pCmdUI)

②、ON_UPDATE_COMMAND_UI_RANGE(nID,nLastID,pfn)
该函数可以处理一组[命令用户接口对象]的外观;
其中函数的原型如下:
afx_msg void pfn(CCmdUI* pCmdUI)
重要:
CCmdUI 中的 m_nID 成员表示不同的 ID,因此可以利用它来进行区别处理。

3、控件的通知消息
从控件和子窗口发送到父窗口的WM_COMMAND通知消息(即在发送命令消息中加入控件的通知码)。
注意:在 Window9x 新控件中不再传送WM_COMMAND通知消息,而是发送 WM_NOTIFY 消息,但为了兼容,旧有的控件还是传送WM_COMMAND消息。
例如:
CEdit控件向父窗口发送 EN_CHANGE 通知代码的WM_COMMAND消息。
注意:框架像传送其它 WM_ 消息一样传送通知消息,但有一个例外,即由 [按钮] 控件发送的 BN_CLICKED 通知消息,被作为命令消息特别处理。
㈠、WM_COMMAND 其所对应的宏如下:
①、ON_CONTROL(通知码, nID,fn)
②、ON_CONTROL_RANGE(通知码, nFirstID,nEndID,fn)
注意:
这两个宏的应用和 ON_COMMAND、ON_COMMAND_RANGE相同,所不同的是在宏前面加入[通知码]。
注意:可以根据不同的控件的[通知码]派生出特定的宏,其所派生的宏一般为在 [通知码] 前面加上 ON_。
㈡、WM_NOTIFY 其所对应的宏如下:
①、ON_NOTIFY(通知码, nID,fn)
其中函数的原型如下:
afx_msg void fn(NMHDR* pNotifyStruct,LRESULT* result)
其中结构:
typedef struct tagNMHDR {
HWND hwndFrom; //发送通知消息的控件的句柄;
UINT idFrom; //发送通知消息的控件的 ID;
UINT code; //通知码;
} NMHDR;

②、ON_NOTIFY_EX(通知码, nID,fn)
表示一个消息在多个对象的成员函数中进行处理。
其中函数的原型如下:
afx_msg BOOL fn(UINT nID,NMHDR* pNotifyStruct,LRESULT* result)
说明:
它必须返回 BOOL 类型的数值,其意义和 ON_COMMAND_EX 相同。

③、ON_NOTIFY_RANGE(通知码, nFirstID,nEnd,fn)
表示多个控件的通知消息在同一个函数中进行处理。
其中函数的原型如下:
afx_msg void fn(UINT nID,NMHDR* pNotifyStruct,LRESULT* result)
说明:
其意义和ON_COMMAND_RANGE相同。

4、反射消息处理
父窗口在处理控件窗口的通知消息WM_CTLCOLOR、WM_COMMAND、WM_NOTIFY时,会把该消息转化为反射消息,并转交给控件子窗口处理,只有在控件子窗体不处理该消息时,父窗口才有机会处理。
注意:在类的属性对话框中的消息页面可查反射消息(前面有"="标志)
①、WM_CTLCOLOR_REFLECT反射消息
其所对应的宏如下:
ON_WM_CTLCOLOR_REFLECT()
反射消息函数的原型:
HBRUSH class_name∷CtlColor(CDC* pDC,UINT nCtlColor)
{
return NULL;
}
该函数用来重置控件的顔色;注意:必须 return CBrush才有效。


5、自定义的窗口消息
自定义窗口消息的消息标志都大于WM_USER(至少是WM_USER+100,因为许多控件都使用这一范围的WM_USER消息)
使用自定义的消息分为二步:
①、在 Resource.h 中定义消息标记
#define WM_MYMSG (WM_USER+1000)

②、在消息映射表中加入消息映射宏
BEGIN_MESSAGE_MAP()

ON_MESSAGE(WM_MYMSG,fn)
END_MESSAGE_MAP()
说明:
其对应的宏为 ON_MESSAGE(),其成员函数的原型为:
afx_msg LRESULT fn(WPARAM,LPARAM)

6、登记消息
①、在系统中注册并获取一个登记消息的消息标记
UINT RegisterWindowMessage(LPCTSTR)
说明:
通过 API 函数来注册消息标记,其中 LPCTSTR 为用户的任意字符串。例如:
UINT WM_MYMSG=RegisterWindowMessage("MYMSG");
其中 WM_MYMSG 是自定义无符号整型的消息标记。

②、在消息映射表中加入消息映射宏
BEGIN_MESSAGE_MAP()

ON_REGISTERED_MESSAGE(WM_MYMSG,fn)
END_MESSAGE_MAP()
说明:
其对应的宏为 ON_REGISTERED_MESSAGE(),其成员函数的原型为:
afx_msg LRESULT fn(WPARAM,LPARAM)
注意:登记消息可以实现跨进程的数据通讯。

7、线程消息
只有继承自CWinThread类才能允许处理线程消息。
①、定义线程的消息标记
有两种方法:
(1)、使用自定义的消息标记,即:WM_USER;
(2)、使用登记的消息标记,即:RegisterWindowMessage;

②、在CWinThread继承类的消息映射表中添加宏
ON_THREAD_MESSAGE(消息标记,fn) //自定义的消息;
ON_REGISTERED_THREAD_MESSAGE(消息标记,fn) //登记的 //消息
③、其函数的原型如下:
afx_msg void fn(WPARAM wPARAM,LPARAM lParam)

④、引发线程消息
线程消息的引发必须调用 CWinThread 类的PostThreadMessage将消息投递到线程消息队列中。
注意:可以通过 AfxGetApp() 函数获取一个全局的应用对象。
PostThreadMessage(UINT,WPARAM,LPARAM)

8、WM_COPYDATA
操作系统维护一块内存来管理 WM_COPYDATA 消息,该消息主要用于跨进程传递数据,传递的数据量达到 232。
①、定义一个 COPYDATASTRUCT 数据结构
typedef struct tagCOPYDATASTRUCT {
DWORD dwData; //自定义的特殊数据;
DWORD cbData; //以字节为单位的 lpData 的大小;
PVOID lpData; //传送的数据内存块的指针;
} COPYDATASTRUCT;

②、其所对应的宏
ON_WM_COPYDATA()

③、其所对应的函数的原型
afx_msg BOOL OnCopyData(CWnd*,COPYDATASTRUCT*)
说明:
CWnd*:发送该消息的窗口的指针;


9、投递和发送消息
通过向一个窗体投递或发送消息,可以间接地驱动窗体的消息过程。
投递(PostMessage):将消息放到线程的消息队列中,然后不等线程处理该消息就直接返回到调用方。
发送(SendMessage):当一个线程向目标线程发送消息时,该线程要一直等待,直到目标线程处理了该消息为止。
①、投递消息
BOOL CWnd∷PostMessage(UINT,WPARAM=0,LPARAM=0)
说明:
CWnd:目标窗口;
该函数将一条消息放入到应用程序的消息队列,然后不等窗口处理就直接返回。

②、发送消息
LRESULT CWnd∷SendMessage(UINT,WPARAM=0,LPARAM=0)
说明:
CWnd:目标窗口;
该函数将一条消息放入到应用程序的消息队列,等待窗口处理后才返回。
为了避免线程陷入永久等待状态,可以用SendMessageTimeout代替SendMessage:
LRESULT SendMessageTimeout(HWND,UINT,WPARAM,LPARAM,UINT,UINT,PDWORD_PTR)
说明:
HWND:窗口句柄;
UINT:消息发送的选项,为SMTO_BLOCK时,可以防止线程无 限等待,即根据一定的超时值返回。
UINT:超时,以毫秒为单位;
PDWORD_PTR:返回值;
注意:CWnd没有对该函数的包装。

③、投递和发送消息
BOOL CWnd∷SendNotifyMessage(UINT,WPARAM,LPARAM)
说明:
CWnd:目标窗口;
该消息具有SendMessage和PostMessage两种功能:
当目标窗口和发送窗口为同一个线程时,则相当于SendMessage的功能;否则当不为同一个线程时,则为PostMessage的功能。

6-1、投递和发送 WM_XXX 消息
在发送标准的 WINDOW 消息时,只要将该消息的 ID、wParam、lParam参数放在 SendMessage()和PostMessage()函数的相应位置即可。

6-2、投递和发送命令消息和控件的通知消息
在投递和发送命令消息时,消息的 ID为 WM_COMMADN,而对于不同的菜单项、加速键、控件则wParam、lParam的取值不同。
wParam分成低、高两部分,低部分为菜单项、加速键、控件的ID。高部分则:
菜单项:0;加速键:1;控件:通知码
lParam:当控件时是控件的句柄,否则为 NULL。

对于wParam参数可以采用自定义宏:
WPARAM MAKEWPARAM(WORD wLow,WORD wHigh)

6-3、投递和发送自定义的窗口消息
在投递和发送自定义的窗口消息时,参数 wParam、lParam 没有特别的涵义,只和普通函数的形参一样进行数据的传递。
注意:
PostMessage 和 SendMessage 是不同的,前者投递后就返回,而后者必须等到消息处理后再返回;所以在参数是 [局部] 或 [临时]时,使用PostMessage函数会引发错误(除非参数使用 指针,则可避免错误),而必须使用SendMessage函数。
6-4、投递和发送注册的窗口消息
和 6-3 基本一样,但它要特别注意的问题是:在跨进程的处理消息时,如果将消息PostMessage、SendMessage到某个进程 A,则必须在进程 B 中获取进程 A 的窗口类名,并通过窗口类名获取窗口的指针,最后再根据指针调用 PostMessage、SendMessage 函数。
注意:在获取窗口的指针时,可以根据窗口类名或窗口的标题。


6-5、投递和发送WM_COPYDATA消息
SendMessage(消息标记,WPARAM,LPARAM)
其中:
消息标记:WM_COPYDATA;
WPARAM:发送该消息的窗口句柄;
LPARAM:COPYDATASTRUCT结构的指针,先通过(LPVOID)进行转换,再通过(LPARAM)进行转换,如下形式:
(LPARAM)(LPVOID)&cds

<think>我们正在讨论MFC自定义消息的定义位置。根据引用[1]和[2],自定义消息通常需要以下步骤:1.定义消息ID:使用#define定义一个消息标识符,通常基于WM_USER或WM_APP加上一个偏移量。2.在接收消息的窗口类中声明消息处理函数(在头文件中)。3.在实现文件中添加消息映射(使用ON_MESSAGE)。4.实现消息处理函数。问题焦点:自定义消息(即消息ID)应该定义在什么位置?根据引用[1]:>程序定义消息ID范围:0x0400-0x7FFF。系统定义的消息都是小于WM_USER的,所以我们定义消息的ID就要大于WM_USER。推荐大100以上#defineWM_MY_MESSAGE(WM_USER+100)引用[2]也提到了在视图类中添加消息映射的步骤,包括在头文件中定义消息函数(但这里指的是声明消息处理函数,而不是定义消息)。因此,自定义消息(例如#defineWM_MY_MESSAGE(WM_USER+100))的定义位置通常是在头文件中。具体来说:-如果该消息只在某个类中使用,可以在该类的头文件中定义。-如果多个类都需要使用该消息,那么应该在一个公共的头文件中定义,以便多个源文件包含。例如:在公共头文件(比如resource.h)或者某个类的头文件中定义:```cpp#defineWM_MY_MESSAGE(WM_USER+100)```然后,在需要发送或接收该消息的源文件中包含这个头文件。引用[3]提到,普通Windows消息和用户自定义的消息只能放到从CWnd派生的类中处理(如CFrameWnd、CView等)。因此,定义消息的头文件需要被这些类的头文件或实现文件包含。总结回答:在MFC中自定义消息消息ID)通常定义在头文件中,可以是:1.单独的头文件(如项目公共头文件)2.使用该消息的窗口类的头文件(如果该消息仅在此类中使用)3.如果多个类需要使用,则定义在公共头文件中这样,在发送消息的地方(可能在其他类或线程中)和接收消息的窗口类中,都可以通过包含同一个头文件来使用相同的消息ID。注意:确保消息ID的唯一性,避免重复定义。下面是一个示例结构:在stdafx.h或一个专门的消息定义头文件(如MyMessages.h)中:```cpp#pragmaonce//自定义消息定义#defineWM_USER_MYMSG(WM_USER+100)```然后在需要处理此消息的窗口类头文件(如MainFrm.h)中:```cpp#include"MyMessages.h"//包含自定义消息定义classCMainFrame:publicCFrameWnd{//...protected:afx_msgLRESULTOnMyMessage(WPARAMwParam,LPARAMlParam);//...};```在MainFrm.cpp中:```cppBEGIN_MESSAGE_MAP(CMainFrame,CFrameWnd)//...其他消息映射ON_MESSAGE(WM_USER_MYMSG,&CMainFrame::OnMyMessage)END_MESSAGE_MAP()LRESULTCMainFrame::OnMyMessage(WPARAMwParam,LPARAMlParam){//处理消息return0;}```在发送消息的地方(比如另一个类中):```cpp#include"MyMessages.h"//确保包含自定义消息定义//获取主框架窗口指针并发送消息CMainFrame*pMainFrame=(CMainFrame*)AfxGetMainWnd();if(pMainFrame){pMainFrame->PostMessage(WM_USER_MYMSG,0,0);}```这样,自定义消息的定义位置就清晰了:在头文件中定义,然后通过包含该头文件来使用。</think>### MFC自定义消息定义位置详解 在MFC中自定义消息的定义位置需遵循以下规范: #### 1. **最佳位置:公共头文件** ```cpp // MyMessages.h (项目公共头文件) #pragma once #define WM_USER_CUSTOM_MSG1 (WM_USER + 100) // 自定义消息1 [^1] #define WM_USER_CUSTOM_MSG2 (WM_USER + 101) // 自定义消息2 ``` **优点**: - 集中管理所有自定义消息ID - 避免重复定义冲突 - 方便多文件引用 #### 2. **特定类头文件中** ```cpp // MainFrm.h (框架类头文件) #pragma once #define WM_UPDATE_STATUS_BAR (WM_USER + 110) // 状态栏更新消息 class CMainFrame : public CFrameWnd { // ... afx_msg LRESULT OnUpdateStatusBar(WPARAM, LPARAM); }; ``` **适用场景**: - 消息仅在该类内部使用 - 避免全局命名污染 #### 3. **实现文件中** ```cpp // ChildView.cpp (视图类实现文件) #define WM_CHILDVIEW_MSG (WM_USER + 120) // 仅本文件可见 BEGIN_MESSAGE_MAP(CChildView, CWnd) ON_MESSAGE(WM_CHILDVIEW_MSG, &CChildView::OnChildMessage) END_MESSAGE_MAP() ``` **注意事项**: - 消息ID对其他文件不可见 - 适用于私有辅助消息 #### ⚠️ 关键规范 1. **命名空间**: ```cpp namespace MyAppMessages { const UINT WM_DATA_READY = WM_USER + 150; // 使用命名空间防止冲突 } ``` 2. **ID范围**: - `WM_USER` (0x0400-0x7FFF):进程内有效 [^1] - `WM_APP` (0x8000-0xBFFF):跨进程可用 ```cpp #define WM_APP_CROSS_PROC (WM_APP + 100) // 跨进程消息 ``` 3. **避免位置**: - **stdafx.h**:可能导致过度包含 - **资源文件(.rc)**:仅用于资源ID #### 完整使用示例 ```cpp // 公共头文件 MyDefines.h #define WM_USER_DATA_RECEIVED (WM_USER + 200) // 数据接收消息 [^1] // 视图类头文件 MyView.h #include "MyDefines.h" class CMyView : public CView { afx_msg LRESULT OnDataReceived(WPARAM, LPARAM); }; // 视图类实现文件 MyView.cpp BEGIN_MESSAGE_MAP(CMyView, CView) ON_MESSAGE(WM_USER_DATA_RECEIVED, &CMyView::OnDataReceived) END_MESSAGE_MAP() // 发送消息处 GetView()->PostMessage(WM_USER_DATA_RECEIVED, 0, (LPARAM)&data); ``` #### 重要原则 1. **唯一性**:确保消息ID在作用域内唯一 2. **可见性**:发送方和接收方需看到相同定义 3. **可维护性**:大型项目推荐使用独立头文件+命名空间 4. **值范围**:自定义消息ID必须 ≥ WM_USER (0x0400) [^1] > 📌 **最佳实践**:在项目早期创建`Messages.h`专门管理自定义消息,避免后续重构成本。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值