设计模式-状态模式

前言

建议在阅读本文前先阅读策略模式这篇文章,虽说状态模式和策略模式的结构几乎相同,但它们的目的、本质完全不同。状态模式的行为是平行的、不可替换的,而策略模式的行为是彼此独立,可以相互替换。读完本章你会有更深刻的认识。

状态模式的定义

当一个对象的内在状态改变时允许改变其行为,这个对象看起来像是改变了其类。

状态模式使用场景

  • 行为随状态改变而改变的场景
    这也是状态模式的根本出发点,例如权限设计,人员级别不同即使执行相同的行为结果也会不同,在这种情况下需要考虑使用状态模式
  • 代码中包含大量与对象状态有关的条件语句。
    状态模式将每一个条件分支(也就是一个状态)放入一个独立的类中,这使得可以根据对象自身的状态而做出相应的行为,从而去除了过多的、重复的 if-else 分支语句。

状态模式的 UML 类图


角色介绍:

  • State:抽象状态类或者状态接口,表示状态下的行为
  • ConcreteStateA、ConcreteStateB:具体状态类,实现了抽象状态类接口,定义本状态下的接口行为。
  • Context:环境类角色,定义客户端感兴趣的接口,负责具体状态的切换。

状态模式示例

使用了状态模式

抽象状态角色
定义了状态的动作

public interface LiftState {
    public void open();

    public void close();

    public void run();

    public void stop();
}

Context 环境类

public class LiftContext {
    // 4 种状态
    public static final OpenningState OPENNING_STATE = new OpenningState();
    public static final ClosingState CLOSEING_STATE = new ClosingState();
    public static final RunningState RUNNING_STATE = new RunningState();
    public static final StoppingState STOPPING_STATE = new StoppingState();
    // 当前状态
    private LiftState mLiftState = STOPPING_STATE;

    private static final LiftContext CONTEXT = new LiftContext();

    public static LiftContext getContext() {
        return CONTEXT;
    }
    // 切换状态
    public void setLiftState(LiftState liftState) {
        this.mLiftState = liftState;
    }

    public void open() {
        mLiftState.open();
    }

    public void close() {
        mLiftState.close();
    }

    public void run() {
        mLiftState.run();
    }

    public void stop() {
        mLiftState.stop();
    }
}

具体状态角色
开门状态

public class OpenningState implements LiftState {
    @Override
    public void open() {
        System.out.println("电梯门开启...");
    }

    @Override
    public void close() {
        LiftContext.getContext().setLiftState(LiftContext.CLOSEING_STATE);
        LiftContext.getContext().close();
    }

    // 门开着,无法运行
    @Override
    public void run() {
        // do nothing
    }

    // 门开着,已经停止
    @Override
    public void stop() {
        // do nothing
    }
}

关门状态

public class ClosingState implements LiftState {
    @Override
    public void open() {
        LiftContext.getContext().setLiftState(LiftContext.OPENNING_STATE);
        LiftContext.getContext().open();
    }

    @Override
    public void close() {
        System.out.println("电梯门关闭...");
    }

    @Override
    public void run() {
        LiftContext.getContext().setLiftState(LiftContext.RUNNING_STATE);
        LiftContext.getContext().run();
    }

    @Override
    public void stop() {
        LiftContext.getContext().setLiftState(LiftContext.STOPPING_STATE);
        LiftContext.getContext().stop();
    }
}

运行状态

public class RunningState implements LiftState {
    // 正在运行,不能开门
    @Override
    public void open() {
        // do nothing
    }

    // 正在运行,门已经关
    @Override
    public void close() {
        // do nothing
    }

    @Override
    public void run() {
        System.out.println("电梯运行中...");
    }

    @Override
    public void stop() {
        LiftContext.getContext().setLiftState(LiftContext.STOPPING_STATE);
        LiftContext.getContext().stop();
    }
}

停止状态

public class StoppingState implements LiftState {
    @Override
    public void open() {
        LiftContext.getContext().setLiftState(LiftContext.OPENNING_STATE);
        LiftContext.getContext().open();
    }

    // 门就关着
    @Override
    public void close() {
        // do nothing
    }

    @Override
    public void run() {
        LiftContext.getContext().setLiftState(LiftContext.RUNNING_STATE);
        LiftContext.getContext().run();
    }

    @Override
    public void stop() {
        System.out.println("电梯停止了...");
    }
}

客户端

public class Client {
    public static void main(String[] args) {
        LiftContext liftContext = LiftContext.getContext();
        liftContext.setLiftState(LiftContext.STOPPING_STATE);
        liftContext.open();
        liftContext.close();
        liftContext.run();
        // 1. 运行情况下按开门按钮
        liftContext.open();
        liftContext.stop();
    }
}

运行结果:

电梯门开启...
电梯门关闭...
电梯运行中...
电梯停止了...

假如不使用状态模式,就通过 if-else 或者 switch-case 完成,代码是下面这个样子:
定义电梯运行接口:

public interface ILift {
    //电梯的4个状态
    public final static int OPENNING_STATE = 1;
    public final static int CLOSING_STATE = 2;
    public final static int RUNNING_STATE = 3;
    public final static int STOPPING_STATE = 4;

    //设置电梯的状态
    public void setState(int state);

    public void open();

    public void close();

    public void run();

    public void stop();
}

具体电梯运行类:

public class Lift implements ILift {
    private int state;

    @Override
    public void setState(int state) {
        this.state = state;
    }

    @Override
    public void open() {
        switch (state) {
            case OPENNING_STATE:
                break;
            case CLOSING_STATE:
                openWithoutLogic();
                setState(OPENNING_STATE);
                break;
            case RUNNING_STATE:
                break;
            case STOPPING_STATE:
                openWithoutLogic();
                setState(OPENNING_STATE);
                break;
        }
    }

    @Override
    public void close() {
        switch (state) {
            case OPENNING_STATE:
                closeWithoutLogic();
                setState(CLOSING_STATE);
                break;
            case CLOSING_STATE:
                break;
            case RUNNING_STATE:
                break;
            case STOPPING_STATE:
                break;
        }
    }

    @Override
    public void run() {
        switch (state) {
            case OPENNING_STATE:
                break;
            case CLOSING_STATE:
                runWithoutLogic();
                setState(CLOSING_STATE);
                break;
            case RUNNING_STATE:
                break;
            case STOPPING_STATE:
                runWithoutLogic();
                setState(CLOSING_STATE);
                break;
        }
    }

    @Override
    public void stop() {
        switch (state) {
            case OPENNING_STATE:
                break;
            case CLOSING_STATE:
                stopWithoutLogic();
                setState(STOPPING_STATE);
                break;
            case RUNNING_STATE:
                stopWithoutLogic();
                setState(STOPPING_STATE);
                break;
            case STOPPING_STATE:
                break;
        }
    }

    private void closeWithoutLogic() {
        System.out.println("电梯门关闭...");
    }

    private void openWithoutLogic() {
        System.out.println("电梯门开启...");
    }

    private void runWithoutLogic() {
        System.out.println("电梯运行中...");
    }

    private void stopWithoutLogic() {
        System.out.println("电梯停止了...");
    }
}

电梯类很明显存在很多问题:

  • 电梯实现类 Lift 过长:主要由于 swtich-case 的判断,如果再增加几种状态,该类会更长。
  • 扩展性差:如果还要增加几种状态,Lift 就需要进行修改,违反了 OCP(开闭原则),但状态模式只需要增加类即可。

总结

状态模式的关键点在于不同的状态下对于同一行为有不同的响应。

状态模式优点:
1.避免了过多的条件语句,使得结构更清晰,提高代码的可维护性
2.可扩展性强,增加状态时只需创建新的状态类,无需修改其他状态类。
3.将特定状态的相关行为封装到一个状态对象中,提供了更好的方法组织与特定状态相关的代码。
状态模式缺点:
1.完全使用状态模式,可能会导致子类会过多

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容