设计模式——状态模式

状态模式

状态模式是一种行为设计模式,让你能在一个对象的内部状态变化时改变其行为,使其看上去就像是改变了自身所属的类一样

其主要思想是程序在任意时刻仅可处于几种有限的状态中。 在任何一个特定状态中, 程序的行为都不相同, 且可瞬间从一个状态切换到另一个状态。 不过, 根据当前状态, 程序可能会切换到另外一种状态, 也可能会保持当前状态不变。 这些数量有限且预先定义的状态切换规则被称为转移。

为了能根据当前状态选择完成相应行为的方法, 绝大部分方法中会包含复杂的条件语句。 修改其转换逻辑可能会涉及到修改所有方法中的状态条件语句, 导致代码的维护工作非常艰难。

状态模式建议为对象的所有可能状态新建一个类, 然后将所有状态的对应行为抽取到这些类中。

状态模式主要解决的是当控制一个对象状态转换的条件表达式过于复杂的情况。把状态的判断逻辑转移到表示不同状态的一系列类当中,可以简化复杂的判断逻辑。

c++ 复制代码
#include <iostream>
#include <string>

using namespace std;

class Context;

// 基状态提供了所有具体的状态的接口
class State{
protected:
    Context *context_;

public:
    virtual ~State(){}

    void set_context(Context *context){
        this->context_ = context;
    }

    virtual void Handle1() = 0;
    virtual void Handle2() = 0;
};

// Context 定义了客户端感兴趣的接口。
// 它还维护对 State 子类实例的引用,该子类实例表示 Context 的当前状态。
class Context{
private:
    State *state_;

public:
    Context(State *state):state_(nullptr){
        this->TransitionTo(state);
    }

    ~Context(){
        delete state_;
    }

    // 允许在运行时更改State对象
    void TransitionTo(State *state){
        cout << "Context: Transition to " << typeid(*state).name() << endl;
        if(this->state_ != nullptr)
            delete this->state_;
        this->state_ = state;
        this->state_->set_context(this);
    }

    // Context将其部分行为委托给当前的State对象
    void Request1(){
        this->state_->Handle1();
    }
    void Request2(){
        this->state_->Handle2();
    }
};

// 具体状态实现与上下文状态相关的各种行为
class ConcreteStateA : public State{
public:
    void Handle1() override;

    void Handle2() override{
        cout << "ConcreteStateA handles request2." << endl;
    }
};

class ConcreteStateB : public State{
public:
    void Handle1() override{
        std::cout << "ConcreteStateB handles request1.\n";
    }

    void Handle2() override{
        std::cout << "ConcreteStateB handles request2.\n";
        std::cout << "ConcreteStateB wants to change the state of the context.\n";

        this->context_->TransitionTo(new ConcreteStateA);
    }
};

// 写在这里是因为里面new了ConcreteStateB
void ConcreteStateA::Handle1(){
    {
        std::cout << "ConcreteStateA handles request1.\n";
        std::cout << "ConcreteStateA wants to change the state of the context.\n";

        this->context_->TransitionTo(new ConcreteStateB);
    }
}

void ClientCode() {
    Context *context = new Context(new ConcreteStateA);
    context->Request1();
    context->Request2();
    delete context;
}

int main() {
    ClientCode();
    return 0;
}

运行结果

复制代码
Context: Transition to 14ConcreteStateA
ConcreteStateA handles request1.
ConcreteStateA wants to change the state of the context.
Context: Transition to 14ConcreteStateB
ConcreteStateB handles request2.
ConcreteStateB wants to change the state of the context.
Context: Transition to 14ConcreteStateA
相关推荐
大熊猫侯佩24 分钟前
浪浪山 iOS 奇遇记:给 APP 裹上 Liquid Glass “琉璃罩”(上集)
ui·界面设计·ios 26·液态玻璃·liquid glass·interactive·glass effect
charlie1145141911 小时前
精读C++设计模式20 —— 结构型设计模式:桥接模式
开发语言·c++·学习·设计模式·桥接模式·c++23·概论
da_vinci_x2 小时前
设计稿秒出“热力图”:AI预测式可用性测试工作流,上线前洞察用户行为
前端·人工智能·ui·设计模式·可用性测试·ux·设计师
大飞pkz7 小时前
【设计模式】访问者模式
开发语言·设计模式·c#·访问者模式
大千UI工场7 小时前
是时候重启了:AIGC将如何重构UI设计师的学习路径与知识体系?
ui·重构·aigc
青草地溪水旁9 小时前
设计模式(C++)详解——策略模式(2)
c++·设计模式·策略模式
笨手笨脚の9 小时前
设计模式-装饰器模式
java·设计模式·装饰器模式·结构型设计模式
我命由我1234511 小时前
Photoshop - Photoshop 工具栏(2)矩形框选工具
经验分享·笔记·学习·ui·photoshop·ps·美工
charlie11451419118 小时前
精读 C++20 设计模式:行为型设计模式 — 访问者模式
c++·学习·设计模式·访问者模式·c++20
开心-开心急了18 小时前
主窗口(QMainWindow)如何放入文本编辑器(QPlainTextEdit)等继承自QWidget的对象--(重构版)
python·ui·pyqt