浅析状态机设计模式

背景

在需求开发的过程中,经常会遇到根据不同的情况作出不同的处理。最直接的就是if...else...。
当场景特别复杂时,判断if就有些力不从心了。加一个场景需要修改大量的代码,这不是一个很好的做法。程序的扩展性特别薄弱。

举个栗子:
当我们给朋友手机打电话的时候,朋友的手机就可能出现几种情况:用户开机,用户关机,用户欠费停机,用户销户等。不同的场景产生不同的结果。

状态模式

状态模式就是用来解决大量不同场景不同行为的模式。
状态模式:允许一个对象在其内部状态改变时改变它的行为,对象看起来似乎修改了它的类。

状态模式包含的角色

  • Context:环境类
  • State:抽象状态类
  • ConcreteState:具体状态类
模式结构

示例

日常生活中,我们遇到的最多的带有状态的对象应该就是电梯了,电梯有开门,关门,运行,停止状态。

public interface ILift {
    // 电梯门开状态
    public void open();

    // 电梯关门状态
    public void close();

    // 电梯移动状态
    public void run();

    // 电梯停止状态
    public void stop();
}

电梯的接口有了,我们可以来看下实现类:

public class Lift implements ILift {
    public void open() {
        System.out.println("lift is opening");
    }

    public void close() {
        System.out.println("lift is closed");
    }

    public void run() {
        System.out.println("lift run up or down");
    }

    public void stop() {
        System.out.println("lift stopped");
    }
}

下面就是电梯类怎么调用的问题?我们知道状态之间是有一定的前提条件的,也就是说状态不能随意转化。

如果电梯处于open状态,那它下一个必然是close关门状态,而不能是run状态;当电梯是close状态,可以是run,open,stop(没按楼层)状态。如下:
Y:代表可以进行状态转化
N:代表不能进行转化
O:代表自己对自己,忽略

open close run stop
open O Y N N
close Y O Y N
run N N O Y
stop Y N Y O

由于我们要进行电梯的状态转换,因此需要定义出电梯的状态:

public interface ILift {
    public final static int OPEN_STATE = 1;
    public final static int CLOSE_STATE = 2;
    public final static int RUN_STATE = 3;
    public final static int STOP_STATE = 4;

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

    // 电梯门开状态
    public void open();

    // 电梯关门状态
    public void close();

    // 电梯移动状态
    public void run();

    // 电梯停止状态
    public void stop();
}

在这边我们把电梯看成了一个对象,里面有状态和状态相应的动作,可以想象后面我们的调用代码基本就是依次调用Lift类的各个函数。

缺点:如果电梯还有其他状态(比如通电,断电状态),则Lift类要修改,不符合开闭原则。

根据状态模式,我们思考把状态看做一个对象,更细粒度的切分电梯这个类,把电梯的每个状态当成一个类,我们来试验下:

public abstract class LiftState {
    //状态转换
    private StageChange stageChange;
    // 电梯门开状态
    public abstract void open();

    // 电梯关门状态
    public abstract void close();

    // 电梯移动状态
    public abstract void run();

    // 电梯停止状态
    public abstract void stop();
}

StageChange是状态转换类,用来进行状态的装换。每个状态都有自身的行为,比如Open状态有一系列自己的行为:

public class LiftOpenState extends LiftState {
    public void open() {
        System.out.println("电梯门已经开启");
    }

    public void close() {
        super.stageChange.setLiftState(StageChange.closeState);
        super.stageChange.getLiftState().close();
        System.out.println("电梯关门");
    }

    public void run() {

    }

    public void stop() {

    }
}

状态模式通过一系列状态自身的流转来达到不同的场景执行不同的动作。

优点

  • 结构清晰
  • 封装性好

状态机

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 初识 我第一次知道状态机,是在大学学习《数字电子技术基础》的时候。一块控制芯片有若干输入数据总线Data_in,一...
    邱simple阅读 25,999评论 2 15
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 13,910评论 1 32
  • javascript设计模式与开发实践 设计模式 每个设计模式我们需要从三点问题入手: 定义 作用 用法与实现 单...
    穿牛仔裤的蚊子阅读 9,793评论 0 13
  • 创建型模式 工厂模式 工厂模式(Factory Pattern)是 Java 中最常用的设计模式之一。这种类型的设...
    隔墙送来秋千影阅读 7,620评论 0 11
  • “哒,哒,哒……”高跟鞋踏在地板的声音在楼道间回响,声控灯在一瞬间亮起。她一边爬着楼梯一边想,丈夫现在是否...
    糖点什么阅读 1,774评论 0 0

友情链接更多精彩内容