mfc中的消息的应用

发表于:2007-07-01来源:作者:点击数: 标签:
mfc中的消息莫是相对于虚函数表来说到底有什么优势?有人说是空间上面有好处。既可以省下长长的虚函数表的内存。比如: 用虚函数的话,window的基类会是 class window { virtual OnSize() = 0; virtual OnMove() = 0; virtual OnContextMenu() = 0; ……等等

 mfc中的消息莫是相对于虚函数表来说到底有什么优势?有人说是空间上面有好处。既可以省下长长的虚函数表的内存。比如:
 用虚函数的话,window的基类会是
 class window
 {
  virtual OnSize() = 0;
  virtual OnMove() = 0;
  virtual OnContextMenu() = 0;
  ……等等
 }
 可想而知,每个由此继承而来的子类将会负担多么大的虚函数表,光是windows自带的就有数十个之多。当然不可取。
 
 可是除此之外,似乎有一种观点是如果机器够快的话就可以采用虚函数的方法了,哪只不不过是ms在当年的无奈之举。嗬嗬,真的是这样吗?非也!
 其一、如果有自定义的消息怎么办?修改基类?
 其二、其实在窗口间互相传递消息可以看成是互相的请求服务,此服务被请求窗体可以响应,也可以不响应。这种灵活性是虚函数方法所无法取得的——如果不支持的话,嗬嗬,对不起,编译时便会报错了。请求方对于被请求方的所有要求仅仅是其是CCmdTarget的子类即可。怎么样,正所谓便宜实惠量又足啊。可就是不太好理解,所以ms提供了许多宏来做这件事情。 
 
 下面看一个具体的例子。这是一个关于treectrl操作的例子。这是一棵表示公司组织的树,上面的节点有
 公司
    ----人事科
          -----张三
          -----李四 
    ----采购 
            -----王二
            -----麻子
 
 即三类节点,公司、科室、员工
 
 当客户右击鼠标时,对于不同的节点弹出的菜单当然是不同的。 
 做法1、判断三种节点的图标,(假定用不同的图标显示不同节点)然后针对不同的情况弹出菜单。这种方法对于显然不妥,有点像经典的
 if ( obj.typeid == ...)
  do something
 else if ( obj.typeid == .. )
  do somthine else ...
 结果就是在treectrl的代码中充满了这样的判断代码
 
 做法2、有class Company, class office, class person,均从CCmdTarget继承。当insert treeitem的时候,产生对象,将其指针和插入的item联系起来,方法可以采用SetItemData(对象的指针),这样的话,处理弹出菜单的方法就变为
 treectrl::OnContextMenu(...)
 {
  CCmdTarget* p = GetItemData(hSelItem);
  p->sendcommand(wm_contextmenu, point)
 }
 这样就ok了。但是有一个缺点,占用太多的内存了!每个item都有多出sizeof(CCmdTarget),有点过分。
 再改进——做法3
 class Company, office, person不变,
 但增加他们的工厂类
 
 class treeitemor
 {
  virtual CCmdTarget* CreateItem() = 0;
 }
 class Companyor
 {
  long m_ID;
  virtual CmdTarget* CreateItem(); //产生Company类
 }
 class office和person同上
 
 弹出菜单变为
 treectrl::OnContextMenu(...)
 {
  treeItemor* ItemCreator = GetItemData(hSelItem);
  CCmdTarget *pCmd = ItemCreator->CreateItem();
  pCmd->SendCommand(WM_CONTEXTMENU, POINT);
 }
 
 呼!总算完了。最后请大家多提意见!

 

  
 


原文转自:http://www.ltesting.net