设计模式与游戏——Command命令模式

解耦(decoupling)

两段相互依赖的代码之间的关系就叫耦合

If two pieces of code are coupled, it means you can’t understand one without understanding the other. If you de-couple them, you can reason about either side independently.

解耦的作用是当修改一段代码时,不会影响到另一段代码

A change to one piece of code doesn’t necessitate a change to another.

一、命令模式(Command)

命令模式将请求发起者与请求执行之间进行解耦。

将一个请求(request)封装成一个对象,因此使用户(users)将客户(clients)以参数的形式执行不同的请求(requests)、队列(queue)、日记请求(log requests)、同时支持撤销操作(undoable operations)。

Encapsulate a request as an object, thereby letting users parameterize clients with different requests, queue or log requests, and support undoable operations.
—《Design Patterns: Elements of Reusable Object-Oriented Software》

命令是具体化方法的调用

A command is a reified method call.
—《Game Programming Patterns》

命令是面向对象中的回调(callbacks)

Commands are an object-oriented replacement for callbacks.
—《Design Patterns: Elements of Reusable Object-Oriented Software》

应用情景:

1、 创建/寻找命令,马上执行

事件触发器相对应事件的处理 之间添加了一层命令层,以至于达到事件 产生(produce)消费(consume) 的解耦。

Command 1
  • 事件触发器触发了一个事件(例如:按下一个按钮)。
  • 根据产生的事件生成或寻找命令
  • 执行命令

好处:

  • 灵活:可以修改事件与执行之间的映射。
  • 代码结构清晰:行为请求的代码与执行的代码进行分离。有时候请求的代码属于较低层,而行为执行的代码属于应用逻辑层,所以将两者分离是有必要的。

举个例子:游戏中的按键修改

在游戏中接受到设备的按键事件的时候,假设我们通过handleButtonEvent函数进行处理上下左右事件。因此我们会写如下代码。

void handleButtonEvent(ButtonType btn)
{
  switch(btn)
  {
    case BUTTON_A: playerMoveLeft();    break;
    case BUTTON_S: playerMoveBack();    break;
    case BUTTON_D: playerMoveRight();   break;
    case BUTTON_W: playerMoveForward(); break;
  }
}

假如直接调用playerMoveLeft()等行为方法,会导致难易实现按键修改、游戏人物左右方向颠倒等功能。

加入中间层的话,就能解除他们之间的耦合。
首先我们需要一个基类Command

class Command
{
public:
  virtual void execute() = 0;
}

其次需要一个子类继承Command,用来实例化具体行为。

class MoveLeftCommand : public Command
{
public:
  virtual void execute() override
  {
    playerMoveLeft();
  }
}

在按钮事件接受处定义每个按钮的命令。

class ButtonHandler
{
public:
  void handleButtonEvent(ButtonType btn);
  ...
}

当修改按键、或者其他操作的时候,只需要通过SetButtonCommand方法就可以修改指定按键的行为了。我们通过实例化不同Command的执行方法,以实现不同的功能。所以命令模式使得程序更为灵活。

void ButtonHandler::handleButtonEvent(ButtonType btn)
{
  switch(btn)
  {
    case BUTTON_A: _buttonA->execute(); break;
    case BUTTON_S: _buttonS->execute(); break;
    case BUTTON_D: _buttonD->execute(); break;
    case BUTTON_W: _buttonW->execute(); break;
  }
}

从上面这个例子中的代码中,可以发现 请求发起 部分的代码与 请求执行 的代码进行了分离。在该例中请求发起应该是属于较低层的逻辑,而请求执行属于较高层的游戏行为逻辑,所以将两部分分离是合理的。从而使得代码逻辑清晰。

2、创建命令,再执行
在第一种情景中,请求的发起者与执行者之间的耦合得到了解决。但是有时候请求的命令的创建与执行之间也会存在耦合。可以通过命令队列进行解耦。

Command 2

游戏中一个游戏对象可以被多个来源控制,例如网络、AI、用户输入等。将所有命令来源收集起来,然后交给命令执行者一次性已收集到的命令。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容

  • 1 场景问题# 1.1 如何开机## 估计有些朋友看到这个标题会非常奇怪,电脑装配好了,如何开机?不就是按下启动按...
    七寸知架构阅读 2,778评论 1 59
  • 工厂模式类似于现实生活中的工厂可以产生大量相似的商品,去做同样的事情,实现同样的效果;这时候需要使用工厂模式。简单...
    舟渔行舟阅读 7,697评论 2 17
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,047评论 25 707
  • 目录 本文的结构如下: 什么是命令模式 为什么要用该模式 模式的结构 代码示例 优点和缺点 适用环境 模式应用 总...
    w1992wishes阅读 1,091评论 2 9
  • 今天早上起床十一点半,浑身无力,只觉得累,虽然生病,但依然不影响我的食欲,我想我可能会一直这样吧,身体越发难受的时...
    麻木木阅读 144评论 0 0