1

参考

  • 《设计模式:可复用面向对象软件的基础 》5.8 State 状态 对象行为型模式
  • 《Android源码设计模式解析与实战》第7章 随遇而安--状态模式

意图

允许一个对象在其内部状态改变时改变它的行为。对象看起来似乎修改了它的类。

别名

状态对象(Objects for States)

适用场景

  • 一个对象的行为取决于它的状态,并且它必须在运行时刻根据状态改变它的行为。
  • 一个操作中含有庞大的多分支的条件语句,且这些操作分支包含这一相同的状态。这个状态通常用一个或多个枚举常量表示。

状态模式将每一个条件分支放入到一个独立的类中,这使得你可以根据对象自身的情况将对象状态作为一个对象,这一对象可以不依赖于其它对象而独立变化,这样可以通过多态来去除多的,重复的if-else等分支语句。

结构

结构

  • Context 环境类,定义客户需要的接口,维护一个State子类的实例,这个实例定义了当前环境的状态。
  • State 抽象状态类或者状态接口,定义一个或者一组接口,便是该状态下的行为。
  • ConcreteStateA,ConcreteStateB 具体状态类,每一个具体的状态类实现抽象State中定义的接口,从而达到不同状态下的不同行为。

优点

State模式将所有与一个特定的状态相关的行为都放入一个状态对象中,它提供了一个更好的方法来组织与特定状态相关的代码,将繁琐的状态判断转换为结构清晰的状态类族,再避免代码膨胀的同时也保证了可扩展性与可维护性。

缺点

增加了系统类和对象的个数。

应用例子1 电视遥控器

描述

电视的状态分为开机和关机状态,开机可以进行频道切换,调整音量等操作,但是这时重复开机键是无效的;关机状态下,频道切换,调整音量,关机都是无效的操作,只有按开机键会生效。

结构

结构

代码实现

/**
 *  电视不同状态下的相同抽象行为接口
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/15  23:30
 */
public interface TvState {
    /**
     * 音量调高
     */
    void turnOn();

    /**
     * 音量调低
     */
    void turnOff();

    /**
     * 下一个频道
     */
    void nextChannel();

    /**
     * 上一个频道
     */
    void preChannel();
}

/**
 *  关机状态下的行为实现
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/15  23:22
 */
public class PowerOffState implements TvState {
    @Override
    public void turnOn() {
       // 没有实现即失灵
    }

    @Override
    public void turnOff() {
        // 没有实现即失灵
    }

    @Override
    public void nextChannel() {
        // 没有实现即失灵
    }

    @Override
    public void preChannel() {
        // 没有实现即失灵
    }
}

/**
 *  开机状态下的行为实现
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/15  23:22
 */
public class PowerOnState implements TvState {
    @Override
    public void turnOn() {
        System.out.println("音量调高");
    }

    @Override
    public void turnOff() {
        System.out.println("音量调低");
    }

    @Override
    public void nextChannel() {
        System.out.println("切换到了下一个频道");
    }

    @Override
    public void preChannel() {
        System.out.println("切换到了上一个频道");
    }
}

/**
 *  电视机遥控器类
 * @author newtrekWang
 * @email  wangjiaxing20160101@gmail.com
 * @time   2018/8/15  23:29
 */
public class TvController {
    private TvState state;

    public void powerOn(){
        this.state = new PowerOnState();
        System.out.println("已开机");
    }

    public void powerOff(){
        this.state = new PowerOffState();
        System.out.println("已关机");
    }

    public void turnOn() {
        state.turnOn();
    }

    public void turnOff() {
        state.turnOff();
    }


    public void nextChannel() {
        state.nextChannel();
    }

    public void preChannel() {
        state.preChannel();
    }

}

应用例子2 Android源码中的WifiDriverState

后更

应用例子3 App中的不同登陆状态下的执行行为

比如应用在已登录状态,点击评论,会正常弹出评论框,而未登录状态下,则是要跳转到登录界面登录后,再正常评论。

所以已登录和未登录状态下的评论行为是不同的,这个就可以用状态模式设计。


WangGavin
35 声望0 粉丝

keep looking