孙鑫VC++讲座笔记-(4)MFC消息映射机制的剖析

2024-03-25 22:32

本文主要是介绍孙鑫VC++讲座笔记-(4)MFC消息映射机制的剖析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

孙鑫VC++讲座笔记-(4)MFC消息映射机制的剖析
 

一,消息映射机制

1,消息响应函数:(例:在CDrawView类响应鼠标左键按下消息)

    1)在头文件(DrawView.h)中声明消息响应函数原型。
        //{{AFX_MSG(CDrawView)   //注释宏
        afx_msg void OnLButtonDown(UINT nFlags, CPoint point);
        //}}AFX_MSG   //注释宏
        说明:
            在注释宏之间的声明在VC中灰色显示。afx_msg宏表示声明的是一个消息响应函数。

    2)在源文件(DrawView.cpp)中进行消息映射。
        BEGIN_MESSAGE_MAP(CDrawView, CView)
        //{{AFX_MSG_MAP(CDrawView)
        ON_WM_LBUTTONDOWN()
        //}}AFX_MSG_MAP
        // Standard printing commands
        ON_COMMAND(ID_FILE_PRINT, CView::OnFilePrint)
        ON_COMMAND(ID_FILE_PRINT_DIRECT, CView::OnFilePrint)
        ON_COMMAND(ID_FILE_PRINT_PREVIEW, CView::OnFilePrintPreview)
        END_MESSAGE_MAP()
    说明:
        在宏BEGIN_MESSAGE_MAP()与END_MESSAGE_MAP()之间进行消息映射。
        宏ON_WM_LBUTTONDOWN()把消息WM_LBUTTONDOWN与它的响应函数OnLButtonDown()相关联。这样一旦有消息的产生,就会自动调用相

关联的消息响应函数去处理。
        宏ON_WM_LBUTTONDOWN()定义如下:
        #define ON_WM_LBUTTONDOWN() \
        { WM_LBUTTONDOWN, 0, 0, 0, AfxSig_vwp, \
        (AFX_PMSG)(AFX_PMSGW)(void (AFX_MSG_CALL CWnd::*)(UINT, CPoint))&OnLButtonDown },

    3)源文件中进行消息响应函数处理。(DrawView.cpp中自动生成OnLButtonDown函数轮廓,如下)
        void CDrawView::OnLButtonDown(UINT nFlags, CPoint point)
        {
        // TODO: Add your message handler code here and/or call default
        CView::OnLButtonDown(nFlags, point);
        }
    说明:
        可见当增加一个消息响应处理,在以上三处进行了修改。可在消息响应函数里添加消息处理代码完成对消息的响应、处理。

2,消息响应的方式:

    1)在基类中针对每种消息做一个虚函数,当子类对消息响应时候,只要在子类中重写这个虚函数即可。缺点:MFC类派生层次很多,如果

在基类对每个消息进行虚函数处理,那么从基类派生的每个子类都将背负一个庞大的虚表,这样浪费内存,故MFC没有采取这中方式而采取消息

映射方式。

    2)消息映射方式:MFC在后台维护了一个句柄和C++对象指针对照表,当收到一个消息后,通过消息结构里资源句柄(查对照表)就可找到

与它对应的一个C++对象指针,然后把这个指针传给基类,基类利用这个指针调用WindowProc()函数对消息进行处理,WindowProc()函数中调用

OnWndMsg()函数,真正的消息路由及处理是由OnWndMsg()函数完成的。由于WindowProc()和OnWndMsg()都是虚函数,而且是用派生类对象指针

调用的,由多态性知最总终调用子类的。在OnWndMsg()函数处理的时候,根据消息种类去查找消息映射,判断所发的消息有没有响应函数,具

体方式是到相关的头文件和源文件中寻找消息响应函数声明(从注释宏//{{AFX_MSG(CDrawView)...//}}AFX_MSG之间寻找),消息映射(从宏

BEGIN_MESSAGE_MAP(...)....END_MESSAGE_MAP()之间寻找),最终找到对应的消息处理函数。当然,如果子类中没有对消息进行处理,则消息

交由基类处理。
    说明:
        virtual LRESULT WindowProc(UINT message, WPARAM wParam, LPARAM lParam);
        virtual BOOL OnWndMsg(UINT message, WPARAM wParam, LPARAM lParam, LRESULT* pResult);


二,有关绘图

1,使用SDK获取DC句柄:
    HDC hdc;
    hdc=::GetDc(m_hWnd);//获取DC句柄
    MoveToEx(hdc,m_ptOrigin.x,m_ptOrigin.y,NULL);
    LineTo(hdc,point.x,point.y);
    ::ReleaseDC(m_hWnd,hdc);//释放DC

2,利用CDC类指针和CWin类成员函数获取DC。
    CDC *pDC=GetDC();
    pDC->MoveTo(m_ptOrigin);
    pDC->LineTo(point);
    ReleaseDC(pDC);

3,利用CClientDC对象。(CClientDC类从CDC类派生来的)
    CClientDC dc(this);
    dc.MoveTo(m_ptOrigin);
    dc.LineTo(point);
    说明:
        The CClientDC class is derived from CDC and takes care of calling the Windows functions GetDC at construction time

and ReleaseDC at destruction time. This means that the device context associated with a CClientDC object is the client area

of a window.

4,利用CWindowDC对象。(CWindowDC类从CDC类派生来的)
    CWindowDC dc(this);//
    dc.MoveTo(m_ptOrigin);
    dc.LineTo(point);
    说明:
        The CWindowDC class is derived from CDC. It calls the Windows functionsGetWindowDC at construction time andReleaseDC

at destruction time. This means that a CWindowDC object accesses the entire screen area of a CWnd (both client and nonclient

areas).

5,GetParent()得到父窗口指针;GetDesktopWindow()得到屏幕窗口指针。

6,利用画笔改变线条颜色和类型:
    CPen pen(PS_DOT,1,RGB(0,255,0));//构造画笔对象
    CClientDC dc(this);CPen *pOldPen=dc.SelectObject(&pen);//将画笔选入DC
    dc.MoveTo(m_ptOrigin);
    dc.LineTo(point);
    dc.SelectObject(pOldPen);//恢复先前的画笔

7,使用画刷(通常利用画刷去填充矩形区域):
    使用单色画刷
        CBrush brush(RGB(255,0,0));//构造画刷对象
        CClientDC dc(this);
        dc.FillRect(CRect(m_ptOrigin,point),&brush);//用指定的画刷去填充矩形区域

    使用位图画刷
        CBitmap bitmap;//构造位图对象(使用前需要初试化)
        bitmap.LoadBitmap(IDB_BITMAP1);//初试化位图对象
        CBrush brush(&bitmap);//构造位图画刷
        CClientDC dc(this);
        dc.FillRect(CRect(m_ptOrigin,point),&brush);//用指定的位图画刷去填充矩形区域

    使用透明画刷
        CBrush *pBrush=CBrush::FromHandle((HBRUSH)GetStockObject(NULL_BRUSH));//获取透明画刷对象指针
        CClientDC dc(this);
        CBrush *pOldBrush=dc.SelectObject(pBrush);//将透明画刷选入DC
        dc.Rectangle(CRect(m_ptOrigin,point));
        dc.SelectObject(pOldBrush);//释放透明画刷
    说明:
        The GetStockObject function retrieves a handle to one of the predefined stock pens, brushes, fonts, or palettes.
        HGDIOBJ GetStockObject(
          int fnObject   // type of stock object
        );

    Returns a pointer to a CBrush object when given a handle to a Windows HBRUSH object.
    static CBrush* PASCAL FromHandle( HBRUSH hBrush );//FromHandle是一个静态方法,故可用CBrush::FromHandle()形式调用。
    注意点:
        1)静态方法不属于某一个具体对象,而属于类本身,在类加载的时候就已经为类静态方法分配了代码去,故可用

CBrush::FromHandle()形式调用。
        2)静态方法中,不能引用非静态的数据成员和方法。
        3)静态数据成员需要在类外单独做初始化,形式如: 变量类型 类名::变量名=初始值;

8,CDC::SetROP2方法:
    int SetROP2( int nDrawMode );
    Sets the current drawing mode.

这篇关于孙鑫VC++讲座笔记-(4)MFC消息映射机制的剖析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/846446

相关文章

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

Spring使用@Retryable实现自动重试机制

《Spring使用@Retryable实现自动重试机制》在微服务架构中,服务之间的调用可能会因为一些暂时性的错误而失败,例如网络波动、数据库连接超时或第三方服务不可用等,在本文中,我们将介绍如何在Sp... 目录引言1. 什么是 @Retryable?2. 如何在 Spring 中使用 @Retryable

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

SpringBoot 自定义消息转换器使用详解

《SpringBoot自定义消息转换器使用详解》本文详细介绍了SpringBoot消息转换器的知识,并通过案例操作演示了如何进行自定义消息转换器的定制开发和使用,感兴趣的朋友一起看看吧... 目录一、前言二、SpringBoot 内容协商介绍2.1 什么是内容协商2.2 内容协商机制深入理解2.2.1 内容

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�