乱砍设计模式之十
最后更新于:2022-04-01 14:37:04
COMMAND 模式——诸葛亮造木牛流马
junguo
Command模式,中文名称是命令模式。该模式的目的是将不同的请求封装成不同的对象,这样可以用来做请求队列,请求日志,以及撤销的操作。该模式的核心是把请求封装成对象,这里的请求有些不好理解,我们还是看完例子后再说这个。先看例子。
这次没找到太好的例子,只好把木牛流马改造一下来说明我们的例子。《三国演义》上介绍诸葛亮造过这样的东西,相当于一个永动机了,理论上是完全不可能的事情。不过诸葛亮是神仙,老罗说他有这种能耐就算有吧。我们来看看要实现的功能,假设我们在屏幕上画出一头牛出来,用鼠标点一下它的嘴,可以喷火出来,再点一下,可以停止;点一下它的眼睛,可以放光,再点一下停止;点一下它的角,持续做顶人状,再点一下停止。(这段写的有些啰嗦,大家理解意思就好了。写到这里,再看,自己描述的是牛魔王的形象,呵呵,不过标题不改了)点击牛的其它部位也会发生其它相应的动作。
考虑一下,如何去做这些点击动作(这里只考虑这些点击动作的实现)。一般来说,结构化的方式,会为各个动作写一个函数,在这些函数里实现这些功能。如果我们的程序不需要扩展,这样写,其实也无所谓。来想想,如果我们需要为所有这些操作加上撤销操作该怎么办呢?我现在维护的系统是这样做的,新添加一个专门维护撤销操作的类,然后把每次操作的内容生成一个该类的对象,保存起来。这样一来,当对一个操作进行修改的时候,就需要对两个类进行处理。是否可以把这样的操作放到一个类中呢?这样修改起来更方便一些。我们可以用Command来实现这样的功能,也就是说把所有的操作都通过继承Command来实现。我不知道如何能说的更好一些。还是看例子吧。
~~~
//提炼一个Command接口
class Command
{
public:
virtual ~Command(){}
virtual void Execute() = 0;
protected:
Command(){}
};
class LightOnCommand : public Command
{
public:
void Execute()
{
//一般来说这里多是借助其它对象来完成,这个例子从简了
cout << "开始发光" << endl;
}
};
class LightOffCommand : public Command
{
public:
void Execute()
{
cout << "停止发光" << endl;
}
};
class FireOnCommand : public Command
{
public:
void Execute()
{
cout << "开始喷火" << endl;
}
};
class FireOffCommand : public Command
{
public:
void Execute()
{
cout << "结束喷火" << endl;
}
};
~~~
有了命令的代码,我们再来看看该如何组织这样的代码。
~~~
//公牛类
class Bulls
{
private:
//Command数组
vector m_Commands;
public:
Bulls()
{
m_Commands.push_back(new LightOnCommand());
m_Commands.push_back(new LightOffCommand());
m_Commands.push_back(new FireOnCommand());
m_Commands.push_back(new FireOffCommand());
}
//点击张开嘴
void ClickMouthOpen()
{
m_Commands[0]->Execute();
}
void ClickMouthClose()
{
m_Commands[1]->Execute();
}
void ClickEyeOpen()
{
m_Commands[2]->Execute();
}
void ClickEyeClose()
{
m_Commands[3]->Execute();
}
};
~~~
这里通过vecotr把需要的Command组织了起来,然后点击张开嘴等操作的时候,通过调用相应的Command来实现。其实到现在,也没看到Command的好处。设想一下,我们需要为操作提供一个撤消功能。该如何实现?没有Command你也许需要再提供一个类来完成该功能,有了Command,我们可以通过以下的方式实现:
~~~
//提炼一个Command接口
class Command
{
public:
virtual ~Command(){}
virtual void Execute() = 0;
//撤消操作
virtual void UnExecute() = 0;
protected:
Command(){}
};
class LightOnCommand : public Command
{
public:
void Execute()
{
//一般来说这里多是借助其它对象来完成,这个例子从简了
cout << "开始发光" << endl;
}
void UnExecute()
{
cout << "撤销发光操作" << endl;
}
};
~~~
在Command中增加一个撤消操作UnExecute,所有的子类实现该功能就可以了。我们再来看看撤消操作的组织。
~~~
//公牛类
class Bulls
{
private:
vector m_Commands;
//添加一个堆栈作为撤消操作的容器
stack m_UndoCommands;
public:
Bulls()
{
m_Commands.push_back(new LightOnCommand());
m_Commands.push_back(new LightOffCommand());
m_Commands.push_back(new FireOnCommand());
m_Commands.push_back(new FireOffCommand());
}
void ClickMouthOpen()
{
m_Commands[0]->Execute();
//每次操作后,把该操作放入堆栈中
m_UndoCommands.push(m_Commands[0]);
}
void ClickMouthClose()
{
m_Commands[1]->Execute();
m_UndoCommands.push(m_Commands[1]);
}
void ClickEyeOpen()
{
m_Commands[2]->Execute();
m_UndoCommands.push(m_Commands[2]);
}
void ClickEyeClose()
{
m_Commands[3]->Execute();
m_UndoCommands.push(m_Commands[3]);
}
void Undo()
{
//撤消通过调用m_UndoCommands来完成
if ( !m_UndoCommands.empty())
{
m_UndoCommands.top()->UnExecute();
m_UndoCommands.pop();
}
}
};
~~~
这里还是比较简单,其实这里你可以再把Redo操作加进去。再看看它的调用:
~~~
int main(int argc, char* argv[])
{
Bulls bulls;
bulls.ClickMouthOpen();
bulls.ClickEyeOpen();
bulls.ClickMouthClose();
bulls.ClickEyeClose();
bulls.Undo();
bulls.Undo();
bulls.Undo();
return 0;
}
~~~
Command是经常要用到的一个模式,一般支持撤消重做的地方都可以用到它。一般的文本,图形编辑器中都能看到它的影子。理解起来其实并不难,关键是如何去用了。
参考书目:
1, 设计模式——可复用面向对象软件的基础(Design Patterns ——Elements of Reusable Object-Oriented Software) Erich Gamma 等著 李英军等译 机械工业出版社
2, Head First Design Patterns(影印版)Freeman等著 东南大学出版社
3, 道法自然——面向对象实践指南 王咏武 王咏刚著 电子工业出版社