设计模式解析三 行为模式三剑客

一. 前言

前一章讲了结构模式三剑客,而为什么会把装饰模式、代理模式和适配器模式称为结构模式三剑客呢,实际上这三个模式在结构模式使用使用最为广泛,而且最容易弄混,因为这三种设计模式非常相像。
实际上行为模式中也有这样三个设计模式,他们使用广泛,但又非常相似,让人迷惑,那就是接下来要讲的三个设计模式:策略模式、状态模式、命令模式。
相信很多java程序开发者应该都看过阿里巴巴Java开发手册,如果没看过的话,也建议看一看,对于自己写代码来说还是很有帮助,其中编程规约第七节有这样一段描述:

阿里巴巴Java开发手册

最初没学习设计模式的时候我就在想,什么是策略模式,状态模式呢?平时看到自己的if else if else 代码就感觉很头疼,感觉代码很乱,有没有更优雅的方式呢?


二. 策略模式

相信我们都应该玩过游戏植物大战僵尸,在这里面有多种多样的植物,还有多种多样的僵尸,我们今天就从这个植物来开始。
我们的每种植物都有自己的攻击方式,这是一种行为,比如豌豆可以吐豌豆进行攻击,比如大嘴花可以直接吃掉面前的僵尸,那么写成代码是怎么样呢?
伪代码:

        if ("绿豌豆") {
            System.out.println("发射绿豌豆");
        } else if ("蓝豌豆") {
            System.out.println("发射蓝豌豆");
        } else if ("大嘴花") {
            System.out.println("一口吃掉");
        }

不论怎么看,这样都不够优雅,那么如果用策略来封装这个攻击算法会如何呢?让我们从头开始来做这个例子,首先是定义我们植物接口:

public interface Plant {
    void attack();
}

绿豌豆:

public class GreenPeas implements Plant {
    
    private int strikingDistance;
    private PlantBehavior behavior;

    public GreenPeas(int strikingDistance, PlantBehavior behavior) {
        this.strikingDistance = strikingDistance;
        this.behavior = behavior;
    }
    @Override
    public void attack() {
        behavior.act(strikingDistance);
    }
    
}

大嘴花:

public class BigMouthFlower implements Plant {
    
    private int strikingDistance;
    private PlantBehavior behavior;

    public BigMouthFlower(int strikingDistance, PlantBehavior behavior) {
        this.strikingDistance = strikingDistance;
        this.behavior = behavior;
    }

    @Override
    public void attack() {
        behavior.act(strikingDistance);
    }
    
}

炸弹:

public class Bomb implements Plant {

    private int strikingDistance;
    private PlantBehavior behavior ;

    public Bomb(int strikingDistance, PlantBehavior behavior) {
        this.strikingDistance = strikingDistance;
        this.behavior = behavior;
    }

    @Override
    public void attack() {
        behavior.act(strikingDistance);
    }

}

然后是我们的行为接口:

public interface PlantBehavior {
    void act(int distance);
}

绿豌豆攻击行为:

public class GreenPeasBehavior implements PlantBehavior {
    @Override
    public void act(int distance) {
        System.out.println("发射绿豌豆");
    }
}

炸弹行为:

public class BombBehavior implements PlantBehavior {
    @Override
    public void act(int distance) {
        System.out.println("爆炸。。。。");
    }
}

大嘴花行为:

public class BigMouthFlowerBehavior implements PlantBehavior {
    @Override
    public void act(int distance) {
        System.out.println("吃掉面前的僵尸");
    }
}

开始工作了:

        Plant peas = new GreenPeas(10, new GreenPeasBehavior());
        Plant bomb = new Bomb(10, new BombBehavior());
        Plant bigMouth = new BigMouthFlower(1, new BigMouthFlowerBehavior());
        peas.attack();
        bomb.attack();
        bigMouth.attack();

输出:

发射绿豌豆
爆炸。。。。
吃掉面前的僵尸

接下来,突然公司说要出一个疯狂版的植物大战僵尸,希望绿豌豆可以爆炸:

        Plant peas = new GreenPeas(10, new BombBehavior());
        peas.attack();

输出:

爆炸。。。。

发现没有,只要是实现了PlantBehavior接口的任意行为,他们都可以随意替换,哪怕让绿豌豆一口吃掉面前的僵尸也行,这就是策略模式了。
我们可以省略if else大段的代码插入到业务代码中,也许大家会问,我还是要根据植物的类型选择不同的攻击行为,没错,不过我们可以利用一些其他模式来帮助我们做这个动作,比如工厂模式。
定义:
策略模式定义一个算法族,分别封装起来,让他们之间可以互相替换,此模式让算法的变化独立于使用算法的客户。

三. 状态模式

状态模式和策略模式非常相似,他们都有根据不同情况得到不同的算法,但状态模式更注重状态之间更迭,比如我们的自动贩卖机,当我们投币成功后,可以选择我们要的商品,然后商品会掉出来,当没有投币的时候,选择商品则不会出来,而库存没有的话,选择商品,商品不会出来,钱会自己退出来,那么进行一次归纳,这个贩卖机实际上有以下三种状态:

  • 等待投币状态
  • 投币完成状态
  • 售磬状态
    这三种状态应该是一种顺序的更迭,先从等待投币,再到投币,再到售罄(当然,没卖光的话就再次进入等待投币状态),首先定义一个状态,状态只会有两个行为,就是点按商品按钮请求得到商品,和投币动作,在不同状态下动作是不同的,状态的转换也不同:
public interface State {
    void insertCoin();
    void get();
}

等待投币状态:

public class NoCoinState implements State {
    private ShopMachine shopMachine;

    public NoCoinState(ShopMachine shopMachine) {
        this.shopMachine = shopMachine;
    }

    @Override
    public void insertCoin() {
        shopMachine.setCurrentState(shopMachine.getHavecoin());
        System.out.println("投币成功,请取商品");
    }

    @Override
    public void get() {
        System.out.println("请投币");
    }
}

投币完成状态:

public class HaveCoinState implements State {

    private ShopMachine shopMachine;

    public HaveCoinState(ShopMachine shopMachine) {
        this.shopMachine = shopMachine;
    }

    @Override
    public void insertCoin() {
        System.out.println("你已经投过币了");
    }

    @Override
    public void get() {
        shopMachine.decreaseStock();
        System.out.println("得到商品");
        if (shopMachine.getStock() > 0) {
            shopMachine.setCurrentState(shopMachine.getNocoin());
        } else {
            shopMachine.setCurrentState(shopMachine.getSoldOut());
        }

    }
}

售磬状态:

public class SoldOutState implements State {

    private ShopMachine shopMachine;

    public SoldOutState(ShopMachine shopMachine) {
        this.shopMachine = shopMachine;
    }

    @Override
    public void insertCoin() {
        System.out.println("商品卖光了,退币");
    }

    @Override
    public void get() {
        System.out.println("商品卖光了");
    }
}

商品售卖机:

public class ShopMachine {

    private int stock;

    private State nocoin;

    private State havecoin;

    private State soldOut;

    private State currentState;

    public ShopMachine(int stock) {
        this.stock = stock;
        this.nocoin = new NoCoinState(this);
        this.havecoin = new HaveCoinState(this);
        this.soldOut = new SoldOutState(this);
        this.currentState = nocoin;
    }

    public void insertCoin() {
        this.currentState.insertCoin();
    }

    public void get() {
        this.currentState.get();
    }

    public void setCurrentState(State currentState) {
        this.currentState = currentState;
    }
    ...
}

然后运行我们的代码:

    public static void main(String[] args) {
        ShopMachine machine = new ShopMachine(2);
        machine.insertCoin();
        machine.get();
        machine.insertCoin();
        machine.get();
        machine.insertCoin();
        machine.get();
    }

输出结果:

投币成功,请取商品
得到商品
投币成功,请取商品
得到商品
商品卖光了,退币
商品卖光了

我们只设置了两个库存,当投币了两次以后就跳到了售磬状态,进入这个状态便不能恢复到其他状态了,只能等运维人员来补货了。

定义:
状态模式允许对象在内部状态改变时改变它的行为,对象看起来好像改变了它的类。

相信大家也看到了,状态模式和策略模式看起来非常相似,上面的策略模式只是封装了植物的攻击行为,下面的状态模式其实也就是封装了投币行为和获取商品的行为。那么区别在哪呢?
区别就是:

  • 状态模式注重状态在内部随着时间流逝的自我改变,调用者不关心现在,将来会是什么状态,有哪些状态
  • 而策略模式,需要调用者明确指定使用哪种策略

看起来策略模式更加灵活,但状态模式在有些固定状态更迭流程的场景下可以运行的更简单。

四. 命令模式

家家户户都少不了电视,我很喜欢看电视,现在都流行使用机顶盒,但存在一个问题,使用机顶盒的遥控器只能遥控机顶盒,当想关闭电视,直接控制电视的音量就不行了,这时候大部分机顶盒遥控器都为我们提供了一个学习区,可以操作学习区的按钮学习我们电视遥控器上的功能,比如待机按钮,比如声音调节,我们可以任意学习,那么这是怎么做到的呢?
它的代码怎么写的我不知道,但是如果从java的角度来说,这时候不得不提到命令模式,命令模式允许我们对指令进行封装,然后任意的装载指令,首先定义一个遥控器:

public class RemoteController {

    private Commond red;

    private Commond up;

    private Commond down;

    private Commond left;

    private Commond right;

    public void pressRed() {
        red.execute();
    }

    public void pressUp() {
        up.execute();
    }

    public void pressDown() {
        down.execute();
    }

    public void pressLeft() {
        left.execute();
    }

    public void pressRight() {
        right.execute();
    }
    ...
}

定义命令接口:

public interface Commond {
    void execute();
}

定义一个电视对象:

public class Television {
    /**
     * 0 电视处于关机状态
     * 1 电视处于开机状态
     */
    private int state = 0;
    private int vol = 5;
    private int channel = 1;
    
    public Television() {
    }
    public void onOff() {
        if (state == 0) {
            System.out.println("打开电视");
            this.state = 1;
        } else {
            System.out.println("关闭电视");
            this.state = 0;
        }
    }
    public void addVol() {
        this.vol++;
        System.out.println("音量增加到:" + vol);
    }
    public void decreaseVol() {
        this.vol--;
        System.out.println("音量减少到:" + vol);
    }
    public void addChannel() {
        this.channel++;
        System.out.println("频道增加到:" + channel);
    }
    public void decreaseChannel() {
        this.channel--;
        System.out.println("频道减少到:" + channel);
    }
}

定义开机关机命令:

public class CommondOnOff implements Commond {
    
    private Television television ;

    public CommondOnOff(Television television) {
        this.television = television;
    }

    @Override
    public void execute() {
        television.onOff();
    }

}

声音增加减少命令:

public class CommondAddVol implements Commond {
    private Television television ;
    public CommondAddVol(Television television) {
        this.television = television;
    }
    @Override
    public void execute() {
        television.addVol();
    }
}
public class CommondDecreaseVol implements Commond {
    private Television television ;
    public CommondDecreaseVol(Television television) {
        this.television = television;
    }
    @Override
    public void execute() {
        this.television.decreaseVol();
    }
}

频道增减命令:

public class CommondAddChannel implements Commond {
    private Television television ;
    public CommondAddChannel(Television television) {
        this.television = television;
    }
    @Override
    public void execute() {
        television.addChannel();
    }
}
public class CommondDecreaseChannel implements Commond {
    private Television television;
    public CommondDecreaseChannel(Television television) {
        this.television = television;
    }
    @Override
    public void execute() {
        television.decreaseChannel();
    }
}

接下来进行组装:

    public static void main(String[] args) {
        Television tv = new Television();
        RemoteController control = new RemoteController();
        control.setRed(new CommondOnOff(tv));
        control.setUp(new CommondAddVol(tv));
        control.setDown(new CommondDecreaseVol(tv));
        control.setLeft(new CommondDecreaseChannel(tv));
        control.setRight(new CommondAddChannel(tv));
        control.pressRed();
        control.pressUp();
        control.pressDown();
        control.pressRight();
        control.pressLeft();
        control.pressRed();
    }

运行结果:

打开电视
音量增加到:6
音量减少到:5
频道增加到:2
频道减少到:1
关闭电视

这样就实现了一个简单的命令模式,当然这是最简单的。
命令模式的角色分为客户端、执行者、接收者,上面的遥控器对象是执行者和,而电视则是命令接收者。

  • 客户端(Client)负责装载发送命令,也只有发送者才清楚自己要执行的是什么命令
  • 接收者(Receiver)接收命令发来的请求,做实际的工作
  • 而执行者(Invoker)负责执行命令,执行者也不清楚自己执行的是什么命令

命令模式的核心在于将要执行的操作封装成命令,将命令内部操作对执行者透明。
通常我们在使用命令模式可能并不会这么简单,上面的客户端在装载命令后还是要调用遥控器的按钮来执行命令,在实际编程中有些业务场景我们还可以将命令的发送和执行进行解耦,比如我们将命令模式的接受者设计为一个队列,发送者不断将命令加入到队列中,然后设计一个执行者不断去队列中获取命令来执行。

上面这段话有没有很熟悉,仔细思考,我们的线程池不就是这样一种设计嘛?客户端不断把实现了Runnable接口的命令加入到线程池队列中,而执行者ThreadPoolExecutor则不断去队列中获取命令来执行。

同样的,命令模式还可以在命令中增加撤销操作,思考一些我们系统编辑的撤销操作时怎么实现的?

定义:
在软件系统中,“行为请求者”与“行为实现者”通常呈现一种“紧耦合”。但在某些场合,比如要对行为进行“记录、撤销/重做、事务”等处理,这种无法抵御变化的紧耦合是不合适的。在这种情况下,如何将“行为请求者”与“行为实现者”解耦?将一组行为抽象为对象实现二者之间的松耦合。这就是命令模式

五. 结语

讲到这里实际上三种设计模式的不同也已经体现出来了:

  • 策略模式注重对算法的封装和相互替代
  • 状态模式注重向外部请求隐藏对象内部状态流程的自我转换
  • 而命令模式则注重命令的请求和执行的解耦

设计模式的不同往往都体现在使用的意图上,注意不要搞混哦?

下一章回聊到策略模式的孪生兄弟:模板方法模式

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

推荐阅读更多精彩内容

  • 终于知道原来宝玉爱的是黛玉。 爱林姑娘的笑,爱林姑娘的任性,爱林姑娘的才气,最爱的是林姑娘的理解。 人人都道修身齐...
    Bonnie徐丫丫阅读 56评论 0 0
  • 干货-画材篇今天是五四青年,假期刚过完就又迎来了一个周末啦,小伙伴们应该都吃好玩好了吧~喜欢画画的你是不是该继续开...
    王火火爱你阅读 663评论 0 0
  • 番茄工作法流程 什么是番茄工作法?简单说,就是列出你当天要做的事,设置25分钟闹钟,然后从第一件事开始。此外还要有...
    杨荣玲阅读 359评论 0 1
  • 早上起床的时候,发现手机上的日历写的是1,愣了一下才想起,原来已经12月份了。 感觉到有点惶恐,因为16年已经倒数...
    明初的日记本阅读 223评论 0 0