【设计模式系列】行为型模式之Null Object模式

概要
系统中会包含很多行为,有些行为总是存在并需要被执行,而有些行为在某些场合或某种设置下就不希望被执行,通常很多人会通过一些条件对系统行为进行限制来达到目的,但是这样会给本来很简单纯粹的处理增加额外的逻辑,而通过NULL Object模式会提供一个具有代理式的空白行为的对象来解决问题。

目的
通过NULL Object模式提供一个具有代理式的空白行为的对象,从而隐藏一些判断逻辑。

实例
看这样一个例子,系统中有类Action,一些具体行为都从类Action继承来扩展,类Manager会使用Action类型的对象来处理相应的动作。类图和代码如下:
class Action {
public:
     virtual void Execute() = 0;
};
class ConcreteAction : public Action {
public:
     virtual void Execute() {
          ......
     }
};
class Manager {
public:
     Manager(Action* at) {
          mAction = at;
     }
     void DoAction() {
          mAction->Execute();
     }
private:
     Action* mAction;
};

这时发现在某种情况下我们希望DoAction方法中的Action不做任何处理,怎么解决呢?
假设这里所说的某种情况可以用外部的某个变量isAction来进行判断,当值为FALSE时不进行任何处理。很多人会这样去调整代码,首先把isAction通过构造函数传递给Manager类,然后在DoAction中通过判断,当isAction为TRUE时才执行Action:
class Manager {
public:
     Manager(Action* at,  bool isAction) {
          mAction = at;
          mIsAction = isAction;
     }
     void DoAction() {
          if (mAction == TRUE) {
               mAction->Execute();
          }
     }
private:
     Action* mAction;
     bool mIsAction;
};

还有人会觉得应该把isAction的判断加在Action或ConcreteAction类中,让Execute方法自己来判断是否执行。
虽然这样做都可以解决问题,但是却让本来处理很简单职责很单一的Manager或ConcreteAction类增加了额外的逻辑,也让代码看起来不是那么美妙了。看看Null Object模式怎么解决问题吧。
class NullAction : public Action {
public:
     virtual void Execute() {
          // do nothing
     }
};

就是这样,ConcreteAction,Manager和Action类都不需要改,只是追加了NullAction类,而它的Execute方法什么都不做。实际调用的代码如下所示:
Action* at = NULL;
if (isAction == TRUE) {
     at = new ConcreteAction();
} else {
     at = new NULLAction();
}
Manager mng(at);
mng->DoAction();

可见,这样就把isAction相关的逻辑跟Manager或所有Action类都隔离开来了。 

应用
没啥可说的,该用就用。

posted @ 2012-05-24 13:38  MXi4oyu  阅读(213)  评论(0编辑  收藏  举报