设计模式介绍:模板方法(Template Method)模式

来源:互联网 发布:nginx web服务器配置 编辑:程序博客网 时间:2024/05/19 17:51

 


   应用程序开发框架中的一个基本的概念是模板方法( Template Method )模式,它是如此的常见以至于我们在使用时甚至不觉得这是个模式,达到了熟视无睹的程度。

    模板方法模式的一个重要特征是它(模板方法,这里不是模式名,而是指那个我们称之为模板方法模式的方法或函数)的定义在基类中(有时作为一个保护或私有成员函数)并且不能改动——模板方法模式就是“坚持相同的代码”。它调用其它基类函数(就是那些被派生类覆盖的函数)以便完成其工作,但是客户程序员不必直接调用这些函数。

    我们先提供一个简单的例子,然后再到一些常见的应用程序开发框架中找一些现成的例子来给大家参考。

[cpp] view plaincopy在CODE上查看代码片派生到我的代码片
  1. class IUIElement {  
  2. protected:  
  3.     virtual void drawBackground(Painter * painter) = 0;  
  4.     virtual void drawElement(Painter * painter) = 0;  
  5.     virtual void drawForeground(Painter * painter) = 0;  
  6.       
  7. public:  
  8.     void draw(Painter * painter)  
  9.     {  
  10.         drawBackground(painter);  
  11.         drawElement(painter);  
  12.         drawForeground(painter);  
  13.     }  
  14. };  
  15.   
  16.   
  17. class CButton : public IUIElement {  
  18. protected:  
  19.     void drawBackground(Painter * painter)  
  20.     {  
  21.         //draw background  
  22.     }  
  23.     void drawElement(Painter * painter)  
  24.     {  
  25.         //draw icon  
  26.         //draw text  
  27.     }  
  28.     void drawForeground(Painter * painter)  
  29.     {  
  30.         //draw foreground if necessary  
  31.     }  
  32. };  
    上面的代码非常简单,我们在日常的开发中经常会实现类似的类,其中 IUIElement 的 draw() 方法就是模板方法,它调用 drawBackground 、 drawElement 、 drawForeground 来完成 element 的绘制。

    在我使用过的一些开发框架中,有很多模板方法模式应用的例子,举几个来看看。

    先说 Qt ,它是一个完全用 C++ 实现的应用程序框架,目前可以在 Windows 、 Linux 、Android 、iOS、Mac 、塞班等各种平台上使用。QWidget 类是 Qt 中所有可见控件类的基类。它使用了模板方法模式,我们在使用 QWidget 、QLabel 、QListView 等等这些类时,可以重载 paintEvent 、keyPressEvent 等,但是完全不必关心我们重载的这些方法在何处、何时会被调用,而实际上他们是被 QWidget 中的模板方法(比如 render )调用的。下面是一部分被 QWidget 模板方法调用的可以被我们重载的方法(摘自 Qt 源码):

[cpp] view plaincopy在CODE上查看代码片派生到我的代码片
  1. protected:  
  2.     // Event handlers  
  3.     bool event(QEvent *);  
  4.     virtual void mousePressEvent(QMouseEvent *);  
  5.     virtual void mouseReleaseEvent(QMouseEvent *);  
  6.     virtual void mouseDoubleClickEvent(QMouseEvent *);  
  7.     virtual void mouseMoveEvent(QMouseEvent *);  
  8. #ifndef QT_NO_WHEELEVENT  
  9.     virtual void wheelEvent(QWheelEvent *);  
  10. #endif  
  11.     virtual void keyPressEvent(QKeyEvent *);  
  12.     virtual void keyReleaseEvent(QKeyEvent *);  
  13.     virtual void focusInEvent(QFocusEvent *);  
  14.     virtual void focusOutEvent(QFocusEvent *);  
  15.     virtual void enterEvent(QEvent *);  
  16.     virtual void leaveEvent(QEvent *);  
  17.     virtual void paintEvent(QPaintEvent *);  


    我最早接触的 UI 框架是 MFC / ATL ,然后是 WTL 。MFC 中也有大量的模板方法模式应用,不再详述,这里要提一下 WTL ,它里面也用到了模板方法模式,而且是很特别的一种实现。

    用过 WTL 的开发者都知道这个框架大量使用模板,简直非模板无以 WTL 。我说的 WTL 使用模板方法模式的特别之处就在这里,它通过引入模板,减少了对继承的使用频度(我们知道有个问题叫作“脆弱的基类”问题)。比如 CMDIFrameWindowImpl 这个模板类,实现了 OnSize 方法,通过 MESSAGE_HANDLER 这个宏和 WM_SIZE 消息关联起来,其实 MESSAGE_HANDLER 这个宏已经引入了模板方法,只是我们看不到,有兴趣的可以研究 MFC 的头文件或 ATL 的头文件。特别的地方,我们先看下面的代码然后再回过头来分析。

[cpp] view plaincopy在CODE上查看代码片派生到我的代码片
  1. LRESULT OnSize(UINT /*uMsg*/WPARAM wParam, LPARAM /*lParam*/BOOL/*bHandled*/)  
  2. {  
  3.     if(wParam != SIZE_MINIMIZED)  
  4.     {  
  5.         T* pT = static_cast<T*>(this);  
  6.         pT->UpdateLayout();  
  7.     }  
  8.     // message must be handled, otherwise DefFrameProc would resize the client again  
  9.     return 0;  
  10. }  
    上面的代码演示的不是模板方法本身,而是被模板方法调用的被派生类覆盖的方法。在 OnSize 中,又通过 static_cast 将 this 指针转换为模板类 T 的指针,调用 T 的 UpdateLayout 方法来应对 SIZE 的变化。这样通过传入不同的 T 就可以在编译时生成不同功能的多文档窗口。我们知道策略模式( Strategy ),是在运行时选择算法,而 WTL 则把模板方法模式和策略模式结合了起来,达到了一种很奇妙的效果:策略实际上是编译时(请理解模板的用法)已经固定下来的,但我们写代码时却有种动态选择的感觉。
    

    来看看 Android 上的应用开发,View 是所有可见控件的基类,如果我们要实现一个自定义的 View ,那么 onDraw 、 onMeasure 、 onLayout  、 onKeyDown 等方法恐怕是要重载的。这里面也是模板方法模式在起作用,可以去看 Android 应用框架的源码。


    模板方法模式存在一个基本的问题,就是脆弱的基类问题。比如 MFC ,每次新版本发布后,你都有可能要调整你的程序去适应——因为虽然编译通过,但程序的行为可能因为基类的变化而变得不符合预期。我们说组合优于继承,就和这个问题相关。 java 中有接口( interface ),可以强制派生类实现每一个方法,C++ 中没有接口,但有纯虚函数可以模拟接口,我们要尽可能的使用接口、组合,而不是不假思索的使用继承(实现继承)。

0 0
原创粉丝点击