详解状态模式

news/2025/2/9 5:22:24 标签: 设计模式, java, 状态模式

引言

        水有固态、液态、气态三种状态,在不同条件下这三种状态可以相互转化。同样在软件设计中,有些对象也有不同的状态,不同状态的行为不同,状态模式就是用来处理这种情况的。

1.概念

        状态模式(State Pattern):允许一个对象在其内部状态改变时改变它的行为,对象看起来似乎修改了它的类。其别名为状态对象(Objects for States),状态模式是一种对象行为型模式。

2.模式结构

3.模式分析

        Context:环境类,又称为上下文类,它是拥有多种状态的对象。由于环境类的状态存在多样性且在不同状态下对象的行为有所不同,因此将状态独立出去形成单独的状态类。在环境类中维护一个抽象状态类State的实例,这个实例定义当前状态,在具体实现时它是一个State子类的对象。核心代码如下:

java">class Context {

    private State state;//维持一个对抽象状态对象的引用

    private int value;//其他属性值,该属性值的变化可能会导致对象状态发生变化

    //设置状态对象

    public void setState(State state) {

        this.state = state;

    }



    public void request() {

        //其他代码

        state.handle();//调用状态对象的业务方法

        //其他代码

    }

}

        State:抽象状态类,它用于定义一个接口以封装与环境类的一个特定状态相关的行为,在抽象状态类中声明了各种不同状态对应的方法,而在其子类中实现类这些方法,由于不同状态下对象的行为可能不同,因此在不同子类中方法的实现可能存在不同,相同的方法可以写在抽象状态类中。核心代码如下:

java">public abstract class State {

    //声明抽象业务方法,不同的具体状态类可以不同的实现

    public abstract void handle();

}

        ConcreteState:具体状态类,它是抽象状态类的子类,每一个子类实现一个与环境类的一个状态相关的行为,每一个具体状态类对应环境的一个具体状态,不同的具体状态类其行为有所不同。核心代码如下:

java">class ConcreteState extends State {

    public void handle() {

        //方法具体实现代码

    }

}

        有关状态之间的相互转化,可以有两种方式:1.由环境类负责,根据环境类的属性value的值来控制状态的转化。2.由具体状态类负责,将环境类作为参数传入具体状态类的方法中,根据环境类的属性value的值来控制状态的转化。

4.具体实例分析

        为了简化理解,没有使用value属性,而是选择方式1定义方法进行状态转。

        ElevatorState:抽象状态类,是接口,表示电梯的状态,其中电梯有开门和关门两种状态,开门状态下电梯只能停止,关门状态下电梯才可以移动。具体代码如下:

java">//电梯状态

public interface ElevatorState {

    void openDoor();

    void closeDoor();

    void move();

    void stop();

}

        OpenState:电梯的开门状态,实现抽象状态类接口,模拟开门状态的行为:停止,并且开门状态可以向关门状态转化(在环境类实现)。具体代码如下:

java">// 具体状态类:开门状态

class OpenState implements ElevatorState {

    @Override

    public void openDoor() {

        System.out.println("当前状态:开门状态,电梯已经开门");

    }



    @Override

    public void closeDoor() {

        System.out.println("当前状态:开门状态,电梯要关门了");

    }



    @Override

    public void move() {

        System.out.println("当前状态:开门状态,开门状态下不能移动");

    }



    @Override

    public void stop() {

        System.out.println("当前状态:开门状态,开门状态下电梯已经停止");

    }

}

        CloseState:电梯的关门状态,实现抽象状态类接口,模拟关门状态的行为:电梯移动,并且关门状态可以向开门状态转化(在环境类实现)。具体代码如下:

java">// 具体状态类:关门状态

class CloseState implements ElevatorState {

    @Override

    public void openDoor() {

        System.out.println("当前状态:关门状态,电梯要开门了");

    }



    @Override

    public void closeDoor() {

        System.out.println("当前状态:关门状态,电梯已经关门");

    }



    @Override

    public void move() {

        System.out.println("当前状态:关门状态,电梯正在移动");

    }



    @Override

    public void stop() {

        System.out.println("当前状态:关门状态,电梯要停止了");

    }

}

        ElevatorContext:环境类,控制电梯状态的转化,内部维持了电梯的状态的引用,便于记录此时电梯的状态。具体代码如下:

java">class ElevatorContext {

    private ElevatorState elevatorState;//维持一个对抽象状态对象的引用



    public ElevatorContext() {

        elevatorState = new CloseState(); // 初始状态为关门状态

    }



    public void setState(ElevatorState state) {

        this.elevatorState = state;

    }



    public void openDoor() {

        elevatorState = new OpenState();

        elevatorState.openDoor();

    }



    public void closeDoor() {

        elevatorState = new CloseState();

        elevatorState.closeDoor();

    }



    public void move() {

        elevatorState.move();

    }



    public void stop() {

        elevatorState.stop();

    }

}

        Client:客户端,模拟电梯从开门上人到运送到楼层后开门放人的过程。具体代码如下:

java">public class Client {

    public static void main(String[] args) {

        ElevatorContext elevator = new ElevatorContext();

        elevator.openDoor();

        elevator.move();

        elevator.closeDoor();

        elevator.move();

        elevator.stop();

        elevator.openDoor();

    }

}

        运行代码,结果如下:

5.优缺点

        主要优点如下:

        (1)封装了状态的转换规则,在状态模式中可以将状态的转换代码封装在环境类或者具体状态类中,可以对状态转换代码进行集中管理,而不是分散在一个个业务方法中。

        (2)将所有与某个状态有关的行为放到一个类中,只需要注入一个不同的状态对象即可使环境对象拥有不同的行为。

        (3)允许状态转换逻辑与状态对象合成一体,而不是提供一个巨大的条件语句块,状态模式可以让我们避免使用庞大的条件语句来将业务方法和状态转换代码交织在一起。

        (4)可以让多个环境对象共享一个状态对象,从而减少系统中对象的个数。

        主要缺点如下:

        (1)状态模式的使用必然会增加系统中类和对象的个数,导致系统运行开销增大。

        (2)状态模式的结构与实现都较为复杂,如果使用不当将导致程序结构和代码的混乱,增加系统设计的难度。

        (3)状态模式对“开闭原则”的支持并不太好,增加新的状态类需要修改那些负责状态转换的源代码,否则无法转换到新增状态。而且修改某个状态类的行为也需修改对应类的源代码。

6.适用情况

        (1)对象的行为依赖于它的状态(如某些属性值),状态的改变将导致行为的变化。

        (2)在代码中包含大量与对象状态有关的条件语句,这些条件语句的出现,会导致代码的可维护性和灵活性变差,不能方便地增加和删除状态,并且导致客户类与类库之间的耦合增强。


http://www.niftyadmin.cn/n/5845545.html

相关文章

QT 加图标

1.准备一张图标文件 放在源文件目录下新建一个ICON文件夹,将ICO图标文件放进去 2.添加资源 右键项目->add New 3.错误处理 如果加载不出来,一般是ICO文件大小不对,可以设置为24*24试试,如果还不行,可以依次试试其…

Netty初学四 数据载体ByteBuf

一、ByteBuf的结构: 1.结构图: 从上图可以看出,该结构是一个字节容器,里面的数据分为三部分,第一部分是已经丢弃的字节,这部分的数据是无效的,第二部分是可读字节,这部分数据是该结构…

【学术投稿】第五届计算机网络安全与软件工程(CNSSE 2025)

重要信息 官网:www.cnsse.org 时间:2025年2月21-23日 地点:中国-青岛 简介 第五届计算机网络安全与软件工程(CNSSE 2025)将于2025年2月21-23日在中国-青岛举行。CNSSE 2025专注于计算机网络安全、软件工程、信号处…

初始数据结构☞复杂度与泛式

一.时间复杂度 定义: 算法的时间复杂度是一个数学函数,算法中的基本操作的执行次数,为算法的时间复杂度。 O渐进表示方法: 原因: 计算时间复杂度时,我们其实并不一定要计算精确的执行次数,而…

从Oracle 到帆软BI:打造高效数据可视化仪表盘

从Oracle 到帆软BI:打造高效数据可视化仪表盘 在现代数据驱动的环境中,企业需要通过高效的数据可视化来洞察业务趋势和做出决策。本案例将展示如何利用帆软BI 6.1的强大功能,轻松连接到Oracle数据库,实现数据集成,并创…

MR30分布式IO模块:驱动智能制造工厂的工业互联与高效控制新范式

在工业4.0与智能制造浪潮的推动下,传统制造业正经历着从“机械驱动”向“数据驱动”的深刻转型。作为工业数据连接领域的领军者,明达技术凭借其自主研发的MR30分布式IO模块,以创新的技术架构与卓越的性能表现,为全球制造企业构建了…

Scala语言的区块链

以Scala语言的区块链 随着数字货币和去中心化应用的兴起,区块链技术逐渐成为计算机科学与金融科技领域中的一颗耀眼明星。区块链以其去中心化、不可篡改、透明可信等特性,吸引了无数开发者与企业的关注。在众多编程语言中,Scala凭借其独特的…

python学opencv|读取图像(五十九)使用cv2.dilate()函数实现图像膨胀处理

【1】引言 前序学习过程中,已经初步了解了腐蚀带来的图像处理效果,相关文章链接为: python学opencv|读取图像(五十八)使用cv2.erode()函数实现图像腐蚀处理-CSDN博客 腐蚀其实在一定程度上削减了部分像素&#xff0…