【C++设计模式之状态模式:行为型】分析及示例

简介

状态模式(State Pattern)是一种行为型设计模式,它允许对象在内部状态改变时改变其行为,看起来就像是改变了其类。状态模式将对象的状态封装成不同的类,并使得对象在不同状态下有不同的行为。

描述

状态模式通过将每种状态封装成一个独立的类,然后将具体状态类的行为委托给Context类,使得Context类在不同的状态下具有不同的行为。这样,当Context对象的状态发生变化时,它的行为也会随之改变。

原理

状态模式由三个核心组件组成:Context(上下文类)、State(抽象状态类)和ConcreteState(具体状态类)。

  • Context类负责定义切换状态的口,并维护一个对当前状态对象的引用。Context类将具体的状态行为委托给当前状态对象。
  • State类是一个抽象类,定义了具体状态类需要实现的方法,以及在不同状态下Context对象应该具有的行为。
  • ConcreteState类是具体的状态类,实现了State类定义的方法,并根据当前状态下的需求来执行相应的逻辑。

类图

示例

假设有一个电梯系统,其中电梯有三种状态:打开状态(OpenState)、关闭状态(ClosedState)运行状态(RunningState)。当电梯处于不同的状态时,它的行为也不同。

C++示例代码如下:

cpp 复制代码
#include <iostream>

// Context
class Elevator {
public:
    virtual void open() = 0;
    virtual void close() = 0;
    virtual void run() = 0;
};

// State
class ElevatorState {
public:
    virtual void open(Elevator* elevator) = 0;
    virtual void close(Elevator* elevator) = 0;
    virtual void run(Elevator* elevator) = 0;
};

// ConcreteState
class OpenState : public ElevatorState {
public:
    void open(Elevator* elevator) override {
        std::cout << "The elevator is already open." << std::endl;
    }

    void close(Elevator* elevator) override {
        std::cout << "Closing the elevator..." << std::endl;
        elevator->setState(new ClosedState());
    }

    void run(Elevator* elevator) override {
        std::cout << "Cannot run the elevator while it is open." << std::endl;
    }
};

class ClosedState : public ElevatorState {
public:
    void open(Elevator* elevator) override {
        std::cout << "Opening the elevator..." << std::endl;
        elevator->setState(new OpenState());
    }

    void close(Elevator* elevator) override {
        std::cout << "The elevator is already closed." << std::endl;
    }

    void run(Elevator* elevator) override {
        std::cout << "Running the elevator..." << std::endl;
        elevator->setState(new RunningState());
    }
};

class RunningState : public ElevatorState {
public:
    void open(Elevator* elevator) override {
        std::cout << "Cannot open the elevator while it is running." << std::endl;
    }

    void close(Elevator* elevator) override {
        std::cout << "Cannot close the elevator while it is running." << std::endl;
    }

    void run(Elevator* elevator) override {
        std::cout << "The elevator is already running." << std::endl;
    }
};

// Concrete Context
class ElevatorSystem : public Elevator {
public:
    ElevatorSystem() {
        currentState = new ClosedState();
    }

    void setState(ElevatorState* state) {
        delete currentState;
        currentState = state;
    }

    void open() override {
        currentState->open(this);
    }

    void close() override {
        currentState->close(this);
    }

    void run() override {
        currentState->run(this);
    }

private:
    ElevatorState* currentState;
};

// 使用示例
int main() {
    ElevatorSystem system;

    system.open();  // Opening the elevator...
    system.open();  // The elevator is already open.
    system.close(); // Closing the elevator...
    system.run();   // Running the elevator...
    system.close(); // The elevator is already closed.
    system.open();  // Opening the elevator...
    system.run();   // Cannot run the elevator while it is open.
    
    return 0;
}

输出结果

cpp 复制代码
Opening the elevator...
The elevator is already open.
Closing the elevator...
Running the elevator...
The elevator is already closed.
Opening the elevator...
Cannot run the elevator while it is open.

解释

在上述示例中,Elevator类是上下文类(Context),它定义了切换状态的接口,并维护一个对当前状态对象的引用。ElevatorState是抽象状态类(State),它定义了具体状态类需要实现的方法。OpenState、ClosedState和RunningState是具体状态类(ConcreteState),它们分别实现了ElevatorState的方法,并根据当前状态下的需求执行相应的逻辑。

在示例中,创建了一个电梯系统(ElevatorSystem),初始状态为关闭状态(ClosedState)。通过调用ElevatorSystem的open、close和run方法来切换电梯的状态,并根据不同的状态执行相应的操作。

结论

状态模式通过将对象的状态封装成独立的类,并委托给Context类来管理,达到了解耦的目的。当对象的状态发生变化时,它的行为也会相应地发生变化。

状态模式适用于以下情况:

  • 当一个对象的行为取决于其状态,并且需要在运行时根据状态改变行为时,可以使用状态模式。
  • 当一个类有很多状态,并且状态之间的转换非常复杂时,可以使用状态模式来将每个状态的行为解耦合。

状态模式常见的应用场景包括订单状态管理、电梯状态控制、游戏角色状态机等。

相关推荐
deng-c-f11 分钟前
Linux C/C++ 学习日记(26):KCP协议(二):kcp源码分享
c语言·c++·学习·网络编程·kcp
给大佬递杯卡布奇诺11 分钟前
FFmpeg 基本API avformat_open_input函数内部调用流程分析
c++·ffmpeg·音视频
刻BITTER22 分钟前
用CMake 实现U8g2 的 SDL2 模拟环境
c++·stm32·单片机·嵌入式硬件·arduino
三贝勒文子30 分钟前
C++ 多线程实战 14|如何系统性避免死锁
c++
Yupureki38 分钟前
从零开始的C++学习生活 9:stack_queue的入门使用和模板进阶
c语言·数据结构·c++·学习·visual studio
远远远远子39 分钟前
C++-- 内存管理
c++·算法
小年糕是糕手1 小时前
【数据结构】单链表“0”基础知识讲解 + 实战演练
c语言·开发语言·数据结构·c++·学习·算法·链表
Dobby_051 小时前
【Go】C++ 转 Go 第(一)天:环境搭建 Windows + VSCode 远程连接 Linux
linux·运维·c++·vscode·golang
咸鱼爱学习2 小时前
【题解】B2613【深基1.习5】打字速度
数据结构·c++·算法
一匹电信狗2 小时前
【C++】C++风格的类型转换
服务器·开发语言·c++·leetcode·小程序·stl·visual studio