状态模式

简介

Allow an object to alter its behavior when its internal state changes. The object will appear to change its class.
允许对象在内部状态发生改变时改变它的行为,对象看起来好像修改了它的类。

在软件开发过程中,对于某一项操作,可能存在不同的情况。通常处理多情况问题最直接的方式就是使用if...elseswitch...case条件语句进行枚举。但是这种做法对于复杂状态的判断天然存在弊端:条件判断语句过于臃肿,可读性差,且不具备扩展性,维护难度也大。而如果转换思维,将这些不同状态独立起来用各个不同的类进行表示,系统处于哪种情况,直接使用相应的状态类对象进行处理,消除了if...elseswitch...case等冗余语句,代码更有层次性且具备良好扩展力。

状态模式(State Pattern)主要解决的就是当控制一个对象状态的条件表达式过于复杂时的情况。通过把状态的判断逻辑转移到表示不同状态的一系列类中,可以把复杂的判断逻辑简化。

状态模式 中类的行为是由状态决定的,不同的状态下有不同的行为。其意图是让一个对象在其内部改变的时候,其行为也随之改变。

状态模式 核心:状态与行为绑定,不同的状态对应不同的行为。

主要解决

对象的行为依赖于它的状态(属性),并且会根据它的状态改变而改变它的相关行为。

优缺点

优点

  • 结构清晰:将状态独立为类,消除了冗余的if...elseswitch...case语句,使代码更加简洁,提高系统可维护性;
  • 将状态转换显示化:通常的对象内部都是使用数值类型来定义状态,状态的切换是通过赋值进行表现,不够直观;而使用状态类,在切换状态时,是以不同的类进行表示,转换目的更加明确;
  • 状态类职责明确且具备扩展性;

缺点

  • 类膨胀:如果一个事物具备很多状态,则会造成状态类太多;
  • 状态模式的结构与实现都较为复杂,如果使用不当将导致程序结构和代码的混乱;
  • 状态模式对 开闭原则 的支持并不太好,对于可以切换状态的状态模式,增加新的状态类需要修改那些负责状态转换的源代码,否则无法切换到新增状态,而且修改某个状态类的行为也需修改对应类的源代码;

使用场景

  • 行为随状态改变而改变的场景;
  • 一个操作中含有庞大的多分支结构,并且这些分支取决于对象的状态;

模式讲解

首先来看下 状态模式 的通用 UML 类图:

状态模式

从 UML 类图中,我们可以看到,状态模式 主要包含三种角色:

  • 环境类角色(Context):定义客户端需要的接口,内部维护一个当前状态实例,并负责具体状态的切换;
  • 抽象状态角色(State):定义该状态下的行为,可以有一个或多个行为;
  • 具体状态角色(ConcreteState):具体实现该状态对应的行为,并且在需要的情况下进行状态切换;

以下是 状态模式 的通用代码:

class Client {
    public static void main(String[] args) {
        Context context = new Context();
        context.setState(new ConcreteStateB());
        context.handle();
    }

    //抽象状态:State
    interface IState {
        void handle();
    }

    //具体状态类
    static class ConcreteStateA implements IState {
        @Override
        public void handle() {
            //必要时刻需要进行状态切换
            System.out.println("StateA do action");
        }
    }

    //具体状态类
    static class ConcreteStateB implements IState {
        @Override
        public void handle() {
            //必要时刻需要进行状态切换
            System.out.println("StateB do action");
        }
    }

    //环境类
    static class Context {
        private static final IState STATE_A = new ConcreteStateA();
        private static final IState STATE_B = new ConcreteStateB();
        //默认状态A
        private IState mCurrentState = STATE_A;

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

        public void handle() {
            this.mCurrentState.handle();
        }
    }
}

:上面的代码很好地展现了 状态模式 状态分离的好处,代码清晰。不过上面的代码还未能完全展示 状态模式 的全貌,因为不同的状态之间可能存在自动切换的场景(比如手机处于开机状态后就会立即切换到屏幕点亮状态···),但是上面的代码未体现出切换场景功能。我们可以对上面的代码进行修改,使其满足状态切换功能。具体代码如下所示:

class Client {
    public static void main(String[] args) {
        Context context = new Context();
        context.setState(new ConcreteStateA());
        context.handle();
    }

    // 抽象状态:State
    static abstract class State {
        protected Context mContext;

        public void setContext(Context context) {
            this.mContext = context;
        }

        public abstract void handle();
    }

    // 具体状态类
    static class ConcreteStateA extends State {
        @Override
        public void handle() {
            System.out.println("StateA do action");
            // A状态完成后自动切换到B状态
            this.mContext.setState(Context.STATE_B);
            this.mContext.getState().handle();
        }
    }
    ...
    ...
    // 环境类
    static class Context {
        public static final State STATE_A = new ConcreteStateA();
        public static final State STATE_B = new ConcreteStateB();
        // 默认状态A
        private State mCurrentState = STATE_A;
        {
            STATE_A.setContext(this);
            STATE_B.setContext(this);
        }

        public void setState(State state) {
            this.mCurrentState = state;
            this.mCurrentState.setContext(this);
        }

        public State getState() {
            return this.mCurrentState;
        }

        public void handle() {
            this.mCurrentState.handle();
        }
    }
}

主要就是将抽象状态类State由接口改成抽象类,增加对环境类Context的维护,让具体状态ConcreteState可以不必耦合其他具体状态类,而是借由Context间接进行切换功能。

上面的代码中,客户端访问的是 A 状态,但是程序运行后会自动切换到 B 状态。运行结果如下:

StateA do action
StateB do action

举个例子

例子:比如在简书App阅读文章时,觉得文章写的很好,评论收藏两连发。但是如果处于未登录状态,则要先进行登陆,然后再会执行前面的操作。请用代码实现上述逻辑。

分析:上述的例子很简单,就是一个登陆问题:处于登陆情况下,我们就可以做评论,收藏这些行为,否则,跳转到登陆界面,登陆后再继续执行先前的动作。这里涉及的状态有两种:登陆与未登录,行为有两种:评论,收藏。下面我们使用 状态模式 进行实现,代码如下:

class Client {
    public static void main(String[] args) {
        AppContext context = new AppContext();
        context.favorite();
        context.comment("comment: good article.I like it!");
    }

    static class AppContext {
        public static final UserState STATE_LOGIN_IN = new LoginInState();
        public static final UserState STATE_LOGIN_OUT = new LoginOutState();
        private UserState mCurrentState = STATE_LOGIN_OUT;
        {
            STATE_LOGIN_IN.setContext(this);
            STATE_LOGIN_OUT.setContext(this);
        }

        public void setState(UserState state) {
            this.mCurrentState = state;
            this.mCurrentState.setContext(this);
        }

        public UserState getState() {
            return this.mCurrentState;
        }

        public void favorite() {
            this.mCurrentState.favorite();
        }

        public void comment(String comment) {
            this.mCurrentState.comment(comment);
        }
    }

    static abstract class UserState {
        protected AppContext mContext;

        public void setContext(AppContext context) {
            this.mContext = context;
        }

        public abstract void favorite();

        public abstract void comment(String comment);
    }

    static class LoginInState extends UserState {

        @Override
        public void favorite() {
            System.out.println("favorite: save it");
        }

        @Override
        public void comment(String comment) {
            System.out.println(comment);
        }
    }

    static class LoginOutState extends UserState {

        @Override
        public void favorite() {
            this.switch2Login();
            this.mContext.getState().favorite();
        }

        @Override
        public void comment(String comment) {
            this.switch2Login();
            this.mContext.getState().comment(comment);
        }

        private void switch2Login() {
            System.out.println("jump to login interface!");
            this.mContext.setState(this.mContext.STATE_LOGIN_IN);
        }
    }
}

结果如下:

jump to login interface!
favorite: save it
comment: good article.I like it!

与其他模式的比较

  • 状态模式 vs 责任链模式
    状态模式责任链模式 都能消除if分支过多的问题。某些情况下, 状态模式 中的状态可以理解为责任,那么这种情况下,两种模式都可以使用。
    从定义来看,状态模式 强调的是一个对象内在状态的改变,而 责任链模式 强调的是外部节点对象间的改变。
    从其代码实现上来看,他们间最大的区别就是 状态模式 各个状态对象知道自己下一个要进入的状态对象;而 责任链模式 并不清楚其下一个节点处理对象,因为链式组装由客户端负责。

  • 状态模式 vs 策略模式
    状态模式策略模式 的 UML 类图架构几乎完全一样,但他们的应用场景是不一样的。策略模式 多种算法行为择其一都能满足,彼此之间是独立的,用户可自行更换策略算法;而 状态模式 各个状态间是存在相互关系的,彼此之间在一定条件下存在自动切换状态效果,且用户无法指定状态,只能设置初始状态。

参考

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

推荐阅读更多精彩内容

  • 设计模式概述 在学习面向对象七大设计原则时需要注意以下几点:a) 高内聚、低耦合和单一职能的“冲突”实际上,这两者...
    彦帧阅读 3,741评论 0 14
  • 【学习难度:★★★☆☆,使用频率:★★★☆☆】直接出处:状态模式梳理和学习:https://github.com/...
    BruceOuyang阅读 1,165评论 0 2
  • .概述 在软件开发过程中,应用程序可能会根据不同的情况作出不同的处理。最直接的解决方案是将这些所有可能发生的情况全...
    泥孩儿0107阅读 372评论 0 0
  • 1 场景问题# 1.1 实现在线投票## 考虑一个在线投票的应用,要实现控制同一个用户只能投一票,如果一个用户反复...
    七寸知架构阅读 1,939评论 7 53
  • 坚持打卡第9天 练字打卡第6天,今天练字一张半,晚上练习效果不太好,以后在假期期间尽量在上午完成,如果有事情要出去...
    田慧婷阅读 159评论 0 0