当前位置: 首页 > article >正文

23种设计模式-状态(State)设计模式

文章目录

  • 一.什么是状态模式?
  • 二.状态模式的结构
  • 三.状态模式的应用场景
  • 四.状态模式的优缺点
  • 五.状态模式的C++实现
  • 六.状态模式的JAVA实现
  • 七.代码解释
  • 八.总结

类图: 状态设计模式类图

一.什么是状态模式?

状态模式(State Pattern)是一种行为型设计模式,允许对象在其内部状态改变时改变其行为,看起来就像对象改变了其类。状态模式通过将状态封装为独立的类,使得对象的行为能够根据内部状态动态地改变。
 在状态模式中,Context(环境类)持有一个指向State(状态接口)的引用对象,并通过该对象来进行状态切换。具体的状态通过实现状态接口的不同子类来进行封装。

二.状态模式的结构

  • Context(环境类):用于维护当前状态,并在特定操作时切换状态。它持有一个State对象的引用,用来对状态进行操作。
  • State(状态接口):定义了一个接口,声明了每个具体状态需要实现的行为。
  • ConcreteStateA 和 ConcreteStateB(具体状态类):实现State接口的不同子类,表示对象在不同状态下的行为。
    在这里插入图片描述

三.状态模式的应用场景

  1. 对象行为依赖于状态,且状态在运行时变化:如电梯的状态(上升、下降、停止)、游戏角色的状态(跑步、跳跃、静止)。
  2. 条件语句过多:当对象中有大量if-else或switch语句判断状态时,可以考虑使用状态模式进行优化。

四.状态模式的优缺点

  • 优点:

    • 封装状态转换:将状态与相关行为封装在一个类中,便于维护和扩展。
    • 避免条件判断:通过多态替代了条件判断,代码更具可读性和扩展性。
  • 缺点:

    • 类数量增加:每个状态需要一个类,可能会导致类数量较多。

五.状态模式的C++实现

#include <iostream>
#include <memory>

// 前置声明,避免循环引用
class State;

// 环境类(Context)
class Context {
private:
    std::shared_ptr<State> state;  // 当前状态

public:
    // 设置当前状态
    void setState(std::shared_ptr<State> newState) {
        state = newState;
    }

    // 执行请求,调用当前状态的Handle方法
    void request();
};

// 状态接口(State)
class State {
public:
    virtual void handle(Context& context) = 0;  // 定义纯虚函数,具体状态类需实现
};

// 具体状态类 A(ConcreteStateA)
class ConcreteStateA : public State {
public:
    void handle(Context& context) override {
        std::cout << "ConcreteStateA handling request.\n";
        // 状态切换:可以切换到状态 B
        context.setState(std::make_shared<ConcreteStateB>());
    }
};

// 具体状态类 B(ConcreteStateB)
class ConcreteStateB : public State {
public:
    void handle(Context& context) override {
        std::cout << "ConcreteStateB handling request.\n";
        // 状态切换:可以切换回状态 A
        context.setState(std::make_shared<ConcreteStateA>());
    }
};

// 环境类方法的实现
void Context::request() {
    if (state) {
        state->handle(*this);  // 调用当前状态的 handle 方法
    }
}

// 测试代码
int main() {
    Context context;
    // 初始化状态为 A
    context.setState(std::make_shared<ConcreteStateA>());

    // 执行多次请求,测试状态切换
    context.request();  // ConcreteStateA 处理请求,然后切换到 ConcreteStateB
    context.request();  // ConcreteStateB 处理请求,然后切换到 ConcreteStateA
    context.request();  // ConcreteStateA 处理请求,然后切换到 ConcreteStateB

    return 0;
}

六.状态模式的JAVA实现

// 抽象状态类
abstract class State {
    public abstract void Handle(Context context);
}

// 具体状态A
class ConcreteStateA extends State {
    @Override
    public void Handle(Context context) {
        System.out.println("ConcreteStateA: Handling request and transitioning to ConcreteStateB.");
        context.setState(new ConcreteStateB());
    }
}

// 具体状态B
class ConcreteStateB extends State {
    @Override
    public void Handle(Context context) {
        System.out.println("ConcreteStateB: Handling request and transitioning to ConcreteStateA.");
        context.setState(new ConcreteStateA());
    }
}

// 环境类 (Context)
class Context {
    private State state;

    public Context(State state) {
        this.state = state;
    }

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

    public void Request() {
        if (state != null) {
            state.Handle(this);
        }
    }
}

// 客户端代码
public class StatePatternDemo {
    public static void main(String[] args) {
        // 初始状态为 ConcreteStateA
        Context context = new Context(new ConcreteStateA());

        // 调用多次 Request,观察状态切换
        context.Request(); // 从 A 切换到 B
        context.Request(); // 从 B 切换到 A
        context.Request(); // 再次从 A 切换到 B
    }
}

七.代码解释

  • Context 类
    • Context类持有一个指向State的指针(使用智能指针std::shared_ptr来管理状态对象的生命周期)。
    • 它包含一个request()方法,用于请求当前状态处理相应的操作。
    • setState()方法用于切换状态,方便在运行时改变state的引用指向不同的State对象。
  • State 接口
    • State接口中定义了一个纯虚函数handle(),表示不同状态下的行为。
    • Context通过调用state->handle(*this)来执行当前状态的具体行为。
  • ConcreteStateA 和 ConcreteStateB
    • ConcreteStateA和ConcreteStateB是State接口的具体实现类,每个类都定义了各自的handle()方法。
    • 在ConcreteStateA::handle()中,Context的状态被切换为ConcreteStateB,而在ConcreteStateB::handle()中,状态切换回ConcreteStateA。
    • 这种状态切换体现了状态模式的核心思想,即状态内部控制状态转换,客户端不需要关心状态如何切换。
  • 测试代码
    • 在main函数中,创建Context对象context,并将初始状态设为ConcreteStateA。
    • 通过多次调用request()方法,可以看到状态在ConcreteStateA和ConcreteStateB之间切换,体现出对象行为随状态变化而变化的特点。

八.总结

  状态模式是一种非常适合解决对象行为依赖于状态变化的设计模式。通过将状态封装为独立的类,状态模式使得代码更加清晰和易于维护,避免了条件判断语句的冗余和复杂性。每个具体状态类不仅实现了各自的行为,还决定了状态切换的逻辑,从而让状态之间的转换更为灵活。
关键点

  • 状态模式封装了对象内部的状态变化逻辑,使得对象的行为可以动态改变。
  • 通过多态,状态模式避免了条件判断,使代码更具可扩展性。
  • 状态切换由具体状态类控制,客户端只需与Context交互,降低了耦合度。

http://www.kler.cn/a/401351.html

相关文章:

  • 深入探索Python集合(Set)的高效应用:数据处理、性能优化与实际案例分析
  • mysql bin log分析
  • 【金融风控项目-08】:特征构造
  • 重构代码之删除对参数的赋值
  • [代码随想录Day16打卡] 找树左下角的值 路径总和 从中序与后序遍历序列构造二叉树
  • python调用MySql保姆级教程(包会的)
  • 【深度学习|目标跟踪】DeepSort 详解
  • MATLAB深度学习(二)——如何训练一个卷积神经网路
  • 1.1 计算机系统概述
  • 小Q和小S的游戏 | BFS
  • 【C++】第九节:list
  • Ease Monitor 会把基础层,中间件层的监控数据和服务的监控数据打通,从总体的视角提供监控分析
  • 企业供配电及用电一体化微电网能源管理系统
  • 【Linux】多用户协作
  • 深入解析TK技术下视频音频不同步的成因与解决方案
  • 计算机网络(12)介质访问控制
  • 高光谱遥感是什么?高光谱遥感数据如何处理?(基于Matlab和Python多案例解析)从小白到精通
  • 高可用服务器磁盘,如何做磁盘阵列raid5
  • docker容器之间的卷共享
  • Cenos7利用docker部署mysql报错-request canceled while waiting for connection
  • web——upload-labs——第九关——特殊字符::$DATA绕过
  • 关于网络安全攻防演化博弈的研究小议
  • 豆包MarsCode算法题:完美整数
  • Spring Boot图书馆管理系统:疫情下的解决方案
  • Preamble puncture 信号处理技术
  • 24.11.19 web框架