走出MFC子类化的迷宫:子类化,SUBCLASSWINDOW ,MFC消息机制 ---(摘自CSDN论坛)

本文主要是介绍走出MFC子类化的迷宫:子类化,SUBCLASSWINDOW ,MFC消息机制 ---(摘自CSDN论坛),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

许多Windows程序员都是跳过SDK直接进行RAD开发工具[或VC,我想VC应不属于RAD]的学习,有些人可能对子类化机制比较陌生。   
   我们先看看什么是Windows的子类化。Windows给我们或是说给它自己定义了许多丰富的通用控件,如:Edit、ComboBox    、ListBox……等,这些控件功能丰富,能为我们开发工作带来极大方面,试想:我们单单是自己实现一个EDIT控件是多么的艰难!但是,在实际开发中还是有些情况这些标准控件也无能为力,比如:在我们的应用中要求一个EDIT得到老师对学生的评价A、B、C[不要对我说你想用ComboBox实现J],这时,要求在Edit中禁止对其它字母、数字的输入操作,怎么办?EDIT控件本身没有提供这种机制,我们就可以采用子类化很好的解决这类问题。   
   我们知道,每一个Windows窗口[这里是EDIT]都有一个窗口处理函数负责对消息处理,子类化的办法就是用我们自己的消息处理函数来替代窗口原有的、标准的处理函数。当然我们自己的窗口处理函数只是关心那些特定的消息[在这里当然是WM_CHAR了],而其它消息,再发给原来的窗口函数处理。在SDK中的实现方法是调用函数SetWindowLong    :   
   WNDPROC    *    oldWndProc    =    (WNDPROC)SetWindowLong(hWnd,    GWL_WNDPROC,(DWORD)AfxGetAfxWndProc());   
   其中AfxGetAfxWndProc()是我们自己的窗口处理函数,在其中处理过我们感兴趣的消息后就可能通过返回的原窗口处理函数指针oldWndProc来把其它消息按标准方法处理掉,具体做法请查阅相关资料。   
   但到了MFC“时代”,一切都被包装起来了,原来的窗口类注册、窗口函数都不见了[或是说隐身了],我想对于那些“刨根问底”的程序员有兴趣了解在MFC中的子类化机制,本人就自己做的一点“探索”作出总结,希望能给大家点启示。   
   我们先用MFC实现我上面提到的要求:一个只能输入A,B,C的EDIT控件。   
   启动时界面如下:   
      
   输入时就只能输入A、B、C,并且只允许输入一个字母。   
      
   实现方法:   
   先派生一个自己的类CsuperEdit,Ctrl    +    W后,在其中处理WM_CHAR,然后再编辑这个消息处理函数:   
    
   void    CSuperEdit::OnChar(UINT    nChar,    UINT    nRepCnt,    UINT    nFlags)     
   {   
   //    TODO:    Add    your    message    handler    code    here    and/or    call    default   
   TCHAR    ch[20];   
   GetWindowText(ch,20);   
   if    (strlen(ch)    ==    1    &&    (nChar    <=    'C'    &&    nChar    >=    'A'))   
   return;   
   if    (nChar    !=    'A'     
   &&    nChar    !=    'B'   
   &&    nChar    !=    'C'   
   )   
   return;   
    
   CEdit::OnChar(nChar,    nRepCnt,    nFlags);   
   }   
    
   然后再给我们Cprog1Dlg类中加入一个数据成员CsuperEdit    m_edit,在CProg1Dlg::OnInitDialog()中加入:   
   m_edit.SubclassDlgItem(IDC_EDIT1,this);   
   m_edit.SetWindowText("<请输入A、B、C>");   
   并处理EDIT向DIALOG发送的通知消息:EN_SETFOCUS:   
   void    CProg1Dlg::OnSetfocusEdit1()     
   {   
   //    TODO:    Add    your    control    notification    handler    code    here   
   m_edit.SetWindowText("");   
   m_edit.SetFocus();   
   }   
    
   OK,一切搞定!和SDK的子类化方法比起来,这是多么的容易!   
   我们看看MFC背着我们到底做了什么!这里主要解决两个容易让初学者比较疑惑的问题:   
   1、 m_edit只是我们定义的一个C++类对象,为什么通过它调用其成员函数SetWindowText便可以控制我们程序中资源编号为:IDC_EDIT1的控件?   
   2、 CSuperEdit类为什么可以处理WM_CHAR消息?   
    
   大家都知道,控制Windows窗口、控件、资源……都是通过它们的句柄来实现,如   
   HHANDLE、HWND、HDC都是句柄,它表现为一个32位长整形数据,存放于Windows中的特定区域,我们可以把它理解为指向我们想控制的窗口、控件、资源的索引,有了它,我们就可以控制我们想要控制的对象。   
   这里你可以想到为什么多数API函数都有一个参数HWND    hwnd了吧!   
   BOOL    SetWindowText(   
       HWND    hWnd,                    //    handle    to    window    or    control   
       LPCTSTR    lpString        //    title    or    text   
   );   
我们的C++变量m_edit要想控制IDC_EDIT1,也要通过它的句柄,但这又是如何实现的呢?您可能注意到了m_edit.SubclassDlgItem(IDC_EDIT1,this);一句,对了,这就是关键所在!   
   在此处F9设置断点,F5之后,程序到达此处,F11跟入SubclassDlgItem函数:   
   BOOL    CWnd::SubclassDlgItem(UINT    nID,    CWnd*    pParent)   
   {   
   ASSERT(pParent    !=    NULL);   
   ASSERT(::IsWindow(pParent->m_hWnd));   
    
   //    check    for    normal    dialog    control    first   
   HWND    hWndControl    =    ::GetDlgItem(pParent->m_hWnd,    nID);   
   if    (hWndControl    !=    NULL)   
   return    SubclassWindow(hWndControl);   
    
   #ifndef    _AFX_NO_OCC_SUPPORT   
   if    (pParent->m_pCtrlCont    !=    NULL)   
   {   
   //    normal    dialog    control    not    found   
   COleControlSite*    pSite    =    pParent->m_pCtrlCont->FindItem(nID);   
   if    (pSite    !=    NULL)   
   {   
   ASSERT(pSite->m_hWnd    !=    NULL);   
   VERIFY(SubclassWindow(pSite->m_hWnd));   
    
   #ifndef    _AFX_NO_OCC_SUPPORT   
   //    If    the    control    has    reparented    itself    (e.g.,    invisible    control),   
   //    make    sure    that    the    CWnd    gets    properly    wired    to    its    control    site.   
   if    (pParent->m_hWnd    !=    ::GetParent(pSite->m_hWnd))   
   AttachControlSite(pParent);   
   #endif    //!_AFX_NO_OCC_SUPPORT   
    
   return    TRUE;   
   }   
   }   
   #endif   
    
   return    FALSE;        //    control    not    found   
   }   
   代码开始时对传入的父窗口做些检查,然后就是   
   HWND    hWndControl    =    ::GetDlgItem(pParent->m_hWnd,    nID);   
   if    (hWndControl    !=    NULL)   
   return    SubclassWindow(hWndControl);   
   这是关键的代码,先用hWndControl得到我们IDC_EDIT1控件的句柄,然后调用   
   SubclassWindow函数,这个函数是实现的关键,我们来看一下它做了什么:   
    
    
    
   BOOL    CWnd::SubclassWindow(HWND    hWnd)   
   {   
   if    (!Attach(hWnd))   
   return    FALSE;   
    
   //    allow    any    other    subclassing    to    occur   
   PreSubclassWindow();   
    
   //    now    hook    into    the    AFX    WndProc   
   WNDPROC*    lplpfn    =    GetSuperWndProcAddr();   
   WNDPROC    oldWndProc    =    (WNDPROC)::SetWindowLong(hWnd,    GWL_WNDPROC, (DWORD)AfxGetAfxWndProc());   
   ASSERT(oldWndProc    !=    (WNDPROC)AfxGetAfxWndProc());   
    
   if    (*lplpfn    ==    NULL)   
   *lplpfn    =    oldWndProc;        //    the    first    control    of    that    type    created   
   #ifdef    _DEBUG   
   else    if    (*lplpfn    !=    oldWndProc)   
   {   
   TRACE0("Error:    Trying    to    use    SubclassWindow    with    incorrect    CWnd/n");   
   TRACE0("/tderived    class./n");   
   TRACE3("/thWnd    =    $%04X    (nIDC=$%04X)    is    not    a    %hs./n",    (UINT)hWnd,   
   _AfxGetDlgCtrlID(hWnd),    GetRuntimeClass()->m_lpszClassName);   
   ASSERT(FALSE);   
   //    undo    the    subclassing    if    continuing    after    assert   
   ::SetWindowLong(hWnd,    GWL_WNDPROC,    (DWORD)oldWndProc);   
   }   
   #endif   
    
   return    TRUE;   
   }   
    
   函数Attach内部如下:   
   BOOL    CWnd::Attach(HWND    hWndNew)   
   {   
   ASSERT(m_hWnd    ==    NULL);            //    only    attach    once,    detach    on    destroy   
   ASSERT(FromHandlePermanent(hWndNew)    ==    NULL);   
   //    must    not    already    be    in    permanent    map   
    
   if    (hWndNew    ==    NULL)   
   return    FALSE;   
CHandleMap*    pMap    =    afxMapHWND(TRUE);    //    create    map    if    not    exist   
   ASSERT(pMap    !=    NULL);   
    
   pMap->SetPermanent(m_hWnd    =    hWndNew,    this);   
    
   #ifndef    _AFX_NO_OCC_SUPPORT   
   AttachControlSite(pMap);   
   #endif   
    
   return    TRUE;   
   }   
这里要说明的是pMap->SetPermanent(m_hWnd    =    hWndNew,    this);一句,它把我们IDC_EDIT1的句柄赋值给类CsuperEdit的数据成员m_hWnd    [别忘了我们的CsuperEdit类是派生于Cedit的],大家可能现在已经隐约的明白了些什么,不错,在m_edit.SetWindowText("<请输入A、B、C>");中正是通过这个数据成员m_hWnd实现对IDC_EDIT1控制的:   
   void    CWnd::SetWindowText(LPCTSTR    lpszString)   
   {   
   ASSERT(::IsWindow(m_hWnd));   
    
   if    (m_pCtrlSite    ==    NULL)   
   ::SetWindowText(m_hWnd,    lpszString);   
   else   
   m_pCtrlSite->SetWindowText(lpszString);   
   }   
   其它CEdit类的函数也都是围绕    “m_hWnd    +    API函数”    进行包装的。   
   而我们常用的DDX_Control方法说到底也是调用SubclassWindow。   
    
   怎么样?第一个问题的来龙去脉搞明白了吧?   
    
   现在看看第二个问题:CSuperEdit类为什么可以处理WM_CHAR消息?   
   可能有的朋友现在疑惑,虽然通过句柄实现了m_edit对IDC_EDIT的控制,但发送给它的消息照样跑到EDIT的标准处理函数中,对WM_CHAR的处理是如何实现的呢?   
   如果消息照样跑到EDIT的标准处理函数中,那当然是不能处理了!不知您有没有看到在上面的SubclassWindow函数中有这么一小段我加了重点标示:   
   //    now    hook    into    the    AFX    WndProc   
   WNDPROC*    lplpfn    =    GetSuperWndProcAddr();   
   WNDPROC    oldWndProc    =    (WNDPROC)::SetWindowLong(hWnd,    GWL_WNDPROC, (DWORD)AfxGetAfxWndProc());   
   ASSERT(oldWndProc    !=    (WNDPROC)AfxGetAfxWndProc());   
    
   if    (*lplpfn    ==    NULL)   
   *lplpfn    =    oldWndProc;        //    the    first    control    of    that    type    created   
   再和我们开始讲到的SDK中子类化机制联系起来,明白了吧?MFC在这里神不知鬼不觉的搞起偷天换日的勾当!   
   这个AfxGetAfxWndProc()函数是这样的:   
    
    
   WNDPROC    AFXAPI    AfxGetAfxWndProc()   
   {   
   #ifdef    _AFXDLL   
   return    AfxGetModuleState()->m_pfnAfxWndProc;   
   #else   
   return    &AfxWndProc;   
   #endif   
   }   
   读过侯捷先生《深入浅出MFC》的朋友不知还是否记得MFC的命令路由机制正是以这个函数为起点的!   
   这样当程序收到发给Edit的WM_CHAR时,本应调用EDIT标准窗口处理函数,现在被改为调用LRESULT    CALLBACK    AfxWndProc(HWND    hWnd,    UINT    nMsg,    WPARAM    wParam,    LPARAM    lParam)了,然后WM_CHAR消息进行一系列的流窜,最终成功到达我们的处理函数CSuperEdit::OnChar(UINT    nChar,    UINT    nRepCnt,    UINT    nFlags),至于是如何流窜的、怎么到达的请参考《深入浅出MFC》[如果您的书是繁体电子版,请从566页读起]。   
    
   终于,我们走出了FMC子类化的迷宫。  

这篇关于走出MFC子类化的迷宫:子类化,SUBCLASSWINDOW ,MFC消息机制 ---(摘自CSDN论坛)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

一文带你理解Python中import机制与importlib的妙用

《一文带你理解Python中import机制与importlib的妙用》在Python编程的世界里,import语句是开发者最常用的工具之一,它就像一把钥匙,打开了通往各种功能和库的大门,下面就跟随小... 目录一、python import机制概述1.1 import语句的基本用法1.2 模块缓存机制1.

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

Java如何通过反射机制获取数据类对象的属性及方法

《Java如何通过反射机制获取数据类对象的属性及方法》文章介绍了如何使用Java反射机制获取类对象的所有属性及其对应的get、set方法,以及如何通过反射机制实现类对象的实例化,感兴趣的朋友跟随小编一... 目录一、通过反射机制获取类对象的所有属性以及相应的get、set方法1.遍历类对象的所有属性2.获取

MySQL中的锁和MVCC机制解读

《MySQL中的锁和MVCC机制解读》MySQL事务、锁和MVCC机制是确保数据库操作原子性、一致性和隔离性的关键,事务必须遵循ACID原则,锁的类型包括表级锁、行级锁和意向锁,MVCC通过非锁定读和... 目录mysql的锁和MVCC机制事务的概念与ACID特性锁的类型及其工作机制锁的粒度与性能影响多版本

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

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

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

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

JVM 的类初始化机制

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

Java ArrayList扩容机制 (源码解读)

结论:初始长度为10,若所需长度小于1.5倍原长度,则按照1.5倍扩容。若不够用则按照所需长度扩容。 一. 明确类内部重要变量含义         1:数组默认长度         2:这是一个共享的空数组实例,用于明确创建长度为0时的ArrayList ,比如通过 new ArrayList<>(0),ArrayList 内部的数组 elementData 会指向这个 EMPTY_EL