设计模式 -- 状态模式(State Pattern)

状态模式:类的行为基于它的状态改变 属于行为型模式,创建表示各种状态的对象和一个行为随着状态对象改变而改变的 context 对象。在代码中包含大量与对象状态有关的条件语句可以通过此模式将各种具体的状态类抽象出来

介绍
  • 意图:允许对象在内部状态发生改变时改变它的行为,对象看起来好像修改了它的类。
  • 主要解决问题:对象的行为依赖于它的状态(属性),并且可以根据它的状态改变而改变它的相关行为。
  • 关键代码:通常命令模式的接口中只有一个方法。而状态模式的接口中有一个或者多个方法。而且,状态模式的实现类的方法,一般返回值,或者是改变实例变量的值。也就是说,状态模式一般和对象的状态有关。实现类的方法有不同的功能,覆盖接口中的方法。状态模式和命令模式一样,也可以用于消除 if...else 等条件选择语句。
  • 优点:1、封装了转换规则。 2、枚举可能的状态,在枚举状态之前需要确定状态种类。 3、将所有与某个状态有关的行为放到一个类中,并且可以方便地增加新的状态,只需要改变对象状态即可改变对象的行为。 4、允许状态转换逻辑与状态对象合成一体,而不是某一个巨大的条件语句块。5、可以让多个环境对象共享一个状态对象,从而减少系统中对象的个数。
  • 缺点:1、状态模式的使用必然会增加系统类和对象的个数。 2、状态模式的结构与实现都较为复杂,如果使用不当将导致程序结构和代码的混乱。 3、状态模式对"开闭原则"的支持并不太好,对于可以切换状态的状态模式,增加新的状态类需要修改那些负责状态转换的源代码,否则无法切换到新增状态,而且修改某个状态类的行为也需修改对应类的源代码。
示例

以图形状态 三角形、圆形、矩形为例:

  1. 创建一个接口

    java 复制代码
    public interface State { 
        void doAction(Shape shape);
    
        String toStr();
    }
  2. 创建 Shape类。

    java 复制代码
    public class Shape {
    
        private State state;
    
        public State getState() {
            return state;
        }
    
        public void setState(State state) {
            this.state = state;
        }
    }
  3. 创建实现接口的实体类
    三角形状态

    java 复制代码
    public class TriangleState implements State{
        @Override
        public void doAction(Shape shape) {
            shape.setState(this);
        }
        @Override
        public String toStr(){
            return "三角形";
        }
    }

    圆形状态

    java 复制代码
    public class CircularState implements State{
        @Override
        public void doAction(Shape shape) {
            shape.setState(this);
        }
        @Override
        public String toStr(){
            return "圆形";
        }
    }

    矩形状态

    java 复制代码
    public class RectangleState implements State{
        @Override
        public void doAction(Shape shape) {
            shape.setState(this);
        }
        @Override
        public String toStr(){
            return "矩形";
        }
    }
  4. 使用

    java 复制代码
        public static void main(String[] args) {
            Shape shape = new Shape();
            TriangleState triangleState = new TriangleState();
            triangleState.doAction(shape);
            String state =  shape.getState().toStr();
            //三角形
            CircularState circularState = new CircularState();
            circularState.doAction(shape);
            state =  shape.getState().toStr();
            //圆形
            RectangleState rectangleState = new RectangleState();
            rectangleState.doAction(shape);
            state =  shape.getState().toStr();
            //矩形
        }
相关推荐
卡尔特斯2 小时前
Android Kotlin 项目代理配置【详细步骤(可选)】
android·java·kotlin
白鲸开源2 小时前
Ubuntu 22 下 DolphinScheduler 3.x 伪集群部署实录
java·ubuntu·开源
ytadpole2 小时前
Java 25 新特性 更简洁、更高效、更现代
java·后端
纪莫2 小时前
A公司一面:类加载的过程是怎么样的? 双亲委派的优点和缺点? 产生fullGC的情况有哪些? spring的动态代理有哪些?区别是什么? 如何排查CPU使用率过高?
java·java面试⑧股
JavaGuide3 小时前
JDK 25(长期支持版) 发布,新特性解读!
java·后端
用户3721574261353 小时前
Java 轻松批量替换 Word 文档文字内容
java
白鲸开源3 小时前
教你数分钟内创建并运行一个 DolphinScheduler Workflow!
java
晨米酱4 小时前
JavaScript 中"对象即函数"设计模式
前端·设计模式
Java中文社群4 小时前
有点意思!Java8后最有用新特性排行榜!
java·后端·面试
代码匠心4 小时前
从零开始学Flink:数据源
java·大数据·后端·flink