设计模式系列篇(十七)——状态模式

What

状态模式(State Pattern),允许一个对象在其内部状态改变的时候改变其行为。这个对象看上去就像是改变了它的类一样。状态模式是一种对象行为型模式。

Why

  1. 状态模式可以通过将事件触发的状态转移和动作执行,拆分到不同的状态类中,来避免分支判断逻辑。
  2. 状态模式可以封装转换规则。
  3. 枚举可能的状态,在枚举状态之前需要确定状态种类。
  4. 将所有与某个状态有关的行为放到一个类中,并且可以方便地增加新的状态,只需要改变对象状态即可改变对象的行为。
  5. 允许状态转换逻辑与状态对象合成一体,而不是某一个巨大的条件语句块。
  6. 可以让多个环境对象共享一个状态对象,从而减少系统中对象的个数。

对于这些优点,大家可以从下面具体的实例中去感受。

When

在以下情况下可以使用状态模式:

  1. 对象的行为依赖于它的状态(属性)并且可以根据它的状态改变而改变它的相关行为。
  2. 代码中包含大量与对象状态有关的条件语句,这些条件语句的出现,会导致代码的可维护性和灵活性变差,不能方便地增加和删除状态,使客户类与类库之间的耦合增强。在这些条件语句中包含了对象的行为,而且这些条件对应于对象的各种状态。

How

用一句话来表述,状态模式把所研究的对象的行为包装在不同的状态对象里,每一个状态对象都属于一个抽象状态类的一个子类。状态模式的意图是让一个对象在其内部状态改变的时候,其行为也随之改变。
《超级马里奥》这款游戏,大家应该都玩过吧。马里奥的状态变换规则如下:

  1. 游戏一开始,小马里奥从天而降,欢乐的蹦哒着,虽然金钱袋空空如也;
  2. 当小马里奥吃掉蘑菇后,小马奥就变身大马里奥,并且可以得到一定金钱作为奖励;
  3. 当小马里奥碰到怪兽时,小马里奥就直接gg了,金钱袋直接变空了;
  4. 当大马里奥碰到怪兽时,大马里奥就一波回到解放前,变成了小马里奥,并扣除一部分金钱。

以上马里奥的状态变化伴随金钱的增加与减少,这个案例特别适合使用状态模式来实现。 下面,我们就来用代码实现下该需求。
首先,我们定义一个接口,实现该接口可以实现不同状态的马里奥。该接口下,定义事件,即吃蘑菇和碰到怪兽。

public interface IMario {
    State getState();

    void obtainMushRoom();

    void meetMonster();
}

public class SmallMario implements IMario {
    private MarioStateMachine marioStateMachine;

    public SmallMario(MarioStateMachine marioStateMachine) {
        this.marioStateMachine = marioStateMachine;
    }

    @Override
    public State getState() {
        return State.SMALL;
    }

    @Override
    public void obtainMushRoom() {
        System.out.println("Small Mario obtain mushroom");
        marioStateMachine.addMoney(100);
        marioStateMachine.setState(new SuperMario(marioStateMachine));
        System.out.println("Add money: " + 100);
        System.out.println("Current state: " + marioStateMachine.getState().getState());
    }

    @Override
    public void meetMonster() {
        System.out.println("Small Mario meets monster");
        marioStateMachine.addMoney(200);
        System.out.println("Sub money: " + 200);
        System.out.println("Current state: " + marioStateMachine.getState().getState());
    }
}

public class SuperMario implements IMario {
    private MarioStateMachine marioStateMachine;

    public SuperMario(MarioStateMachine marioStateMachine) {
        this.marioStateMachine = marioStateMachine;
    }

    @Override
    public State getState() {
        return State.SUPER;
    }

    @Override
    public void obtainMushRoom() {
        System.out.println("Super Mario obtain mush room");
        this.marioStateMachine.addMoney(100);
        System.out.println("add money: " + 100);
    }

    @Override
    public void meetMonster() {
        System.out.println("Super Mario meets monster");
        this.marioStateMachine.setState(new SmallMario(marioStateMachine));
        this.marioStateMachine.subMoney(200);
        System.out.println("sub money: " + 200);
        System.out.println("current state: " + marioStateMachine.getState().getState());
    }
}

接下来,我们定义下状态机类,该类下包含两个成员属性——金钱(money)和状态(state),注意状态(state)变量需要在状态变化时传入下一个状态,就像上面SmallMario中的obtainMushRoom方法,需要调用this.marioStateMachine.setState(new SuperMario(this.marioStateMachine)),这样marioStateMachine.getState()时就会返回更新后的状态。

public class MarioStateMachine {
    private Integer money;
    private IMario state;

    public Integer getMoney() {
        return money;
    }

    public IMario getState() {
        return state;
    }

    public void setState(IMario state) {
        this.state = state;
    }

    public MarioStateMachine() {
        this.money = 0;
        this.state = new SmallMario(this);
    }

    public void addMoney(Integer prize) {
        this.money += prize;
    }

    public void subMoney(Integer punishment) {
        this.money -= punishment;
    }
}

这里可能比较绕,可以好好思考下。

最后,来个测试类:

public class TestMain {
    public static void main(String[] args) {
        MarioStateMachine marioStateMachine = new MarioStateMachine();
        marioStateMachine.getState().obtainMushRoom();
        marioStateMachine.getState().meetMonster();
        marioStateMachine.getState().meetMonster();
        Integer money = marioStateMachine.getMoney();
        System.out.println("Final money: " + money);
    }
}

输出如下:

Small Mario obtain mushroom
Add money: 100
Current state: SUPER
Super Mario meets monster
sub money: 200
current state: SMALL
Small Mario meets monster
Sub money: 200
Current state: SMALL
Final money: 100

代码地址

i-learning

写在最后

如果你觉得我写的文章帮到了你,欢迎点赞、评论、分享、赞赏哦,你们的鼓励是我不断创作的动力~

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