0

0

怎样用C++实现状态模式 基于状态机的上下文转换实现

P粉602998670

P粉602998670

发布时间:2025-08-02 11:32:01

|

711人浏览过

|

来源于php中文网

原创

状态模式在c++中通过封装对象状态为独立类并利用继承多态实现行为变化,其核心是将状态转换逻辑集中于上下文类。1. 定义抽象状态类声明接口;2. 创建具体状态类实现各自行为并在适当时触发状态转换;3. 上下文类持有当前状态并负责状态切换及请求分发。为避免状态爆炸,可采用状态合并、状态表、中间状态、组合状态、策略模式或模板方法模式。在游戏开发中,状态模式适用于管理角色如站立、行走、跑步、跳跃等状态,使状态逻辑模块化。状态的初始化和销毁可通过上下文管理或智能指针自动处理,确保内存安全。

怎样用C++实现状态模式 基于状态机的上下文转换实现

状态模式的核心在于将对象的状态封装成独立的类,并允许对象在内部状态改变时改变它的行为。在C++中,我们可以利用继承和多态来实现这一模式,并通过状态机上下文来管理状态的转换。

怎样用C++实现状态模式 基于状态机的上下文转换实现

解决方案

怎样用C++实现状态模式 基于状态机的上下文转换实现

首先,定义一个抽象状态类,它声明了所有具体状态类需要实现的方法。然后,创建具体状态类,每个类代表对象的一个特定状态,并实现抽象状态类中声明的方法。最后,创建一个上下文类,它持有当前状态的引用,并负责状态之间的转换。

立即学习C++免费学习笔记(深入)”;

#include 
#include 

// 抽象状态类
class State {
public:
    virtual void handle(class Context* context) = 0;
    virtual ~State() {}
};

// 具体状态类 A
class ConcreteStateA : public State {
public:
    void handle(Context* context) override;
};

// 具体状态类 B
class ConcreteStateB : public State {
public:
    void handle(Context* context) override;
};

// 上下文类
class Context {
private:
    State* state;

public:
    Context(State* initialState) : state(initialState) {}

    void setState(State* newState) {
        state = newState;
        std::cout << "State changed to: " << typeid(*state).name() << std::endl; // 打印状态类型,方便调试
        state->handle(this);
    }

    void request() {
        state->handle(this);
    }

    State* getState() const { return state; } // 添加获取状态的接口,方便外部观察
};


void ConcreteStateA::handle(Context* context) {
    std::cout << "ConcreteStateA handles the request." << std::endl;
    // 根据条件转换到状态 B
    context->setState(new ConcreteStateB());
}

void ConcreteStateB::handle(Context* context) {
    std::cout << "ConcreteStateB handles the request." << std::endl;
    // 根据条件转换到状态 A
    context->setState(new ConcreteStateA());
}


int main() {
    Context* context = new Context(new ConcreteStateA());
    context->request();
    context->request();
    context->request();

    delete context->getState(); // 避免内存泄漏,删除最后一个状态
    delete context;

    return 0;
}

代码解释:

怎样用C++实现状态模式 基于状态机的上下文转换实现
  • State
    是抽象状态类,
    handle
    是处理请求的接口。
  • ConcreteStateA
    ConcreteStateB
    是具体状态类,实现了
    handle
    方法,并在方法中根据条件切换状态。
  • Context
    类维护当前状态,并提供
    setState
    方法来切换状态。
    request
    方法将请求委托给当前状态处理。

状态机上下文是关键,它不仅持有当前状态,还负责状态之间的转换逻辑。 状态转换不一定必须在

handle
方法中完成,也可以由外部条件触发,然后调用
context->setState()
进行状态切换。

状态模式的好处是可以将状态相关的行为局部化,使得代码更加清晰和易于维护。 但如果状态过多,类的数量也会增加,可能会增加代码的复杂性。

如何在C++状态模式中避免状态爆炸?

状态爆炸通常发生在状态数量过多,且状态之间的转换关系复杂时。 可以通过以下方法缓解:

  1. 状态合并: 仔细分析状态之间的相似性,如果多个状态的行为非常相似,可以考虑将它们合并成一个状态,并使用内部变量来区分不同的子状态。

  2. 使用状态表: 对于简单的状态转换,可以使用状态表来定义状态之间的转换关系。状态表可以用二维数组或映射表来实现。

  3. 引入中间状态: 如果状态之间的转换路径过长,可以引入中间状态来简化转换过程。

  4. 组合状态: 将多个简单的状态组合成一个复杂的状态,可以减少状态的数量。例如,可以使用组合模式来管理一组状态。

    AVCLabs
    AVCLabs

    AI移除视频背景,100%自动和免费

    下载
  5. 策略模式: 对于某些状态行为,可以使用策略模式来动态选择不同的算法,从而避免为每个状态都实现相同的行为。

  6. 状态模式 + 模板方法模式: 在抽象状态类中使用模板方法模式,将通用的状态处理逻辑放在模板方法中,具体状态类只需要实现特定的状态行为。

选择哪种方法取决于具体的应用场景和状态之间的关系。在设计状态模式时,需要权衡代码的复杂性和可维护性,选择最适合的方案。 状态模式本身是用来解决复杂状态转换问题的,过度使用反而会增加代码的复杂性,需要谨慎。

C++状态模式在游戏开发中的应用案例?

在游戏开发中,状态模式可以用来管理游戏角色的状态,例如站立、行走、跑步、跳跃、攻击、死亡等。每个状态都对应一个具体的状态类,负责处理该状态下的用户输入、动画播放、碰撞检测等。

一个简单的例子是,一个游戏角色在不同的状态下,对键盘输入会有不同的响应。比如,在站立状态下,按下“W”键会进入行走状态;在行走状态下,按下“Shift”键会进入跑步状态;在跑步状态下,按下“空格”键会进入跳跃状态。

// 抽象状态类
class CharacterState {
public:
    virtual void handleInput(class Character* character, Input input) = 0;
    virtual void update(class Character* character, float deltaTime) = 0;
    virtual ~CharacterState() {}
};

// 具体状态类:站立
class StandingState : public CharacterState {
public:
    void handleInput(Character* character, Input input) override {
        if (input == Input::W) {
            character->setState(new WalkingState());
        }
        // ... 其他输入处理
    }
    void update(Character* character, float deltaTime) override {
        // 站立状态下的更新逻辑
    }
};

// 具体状态类:行走
class WalkingState : public CharacterState {
public:
    void handleInput(Character* character, Input input) override {
        if (input == Input::Shift) {
            character->setState(new RunningState());
        } else if (input == Input::S) {
            character->setState(new StandingState()); //停止行走
        }
        // ... 其他输入处理
    }
    void update(Character* character, float deltaTime) override {
        // 行走状态下的更新逻辑
    }
};

// 具体状态类:跑步
class RunningState : public CharacterState {
public:
    void handleInput(Character* character, Input input) override {
        if (input == Input::Space) {
            character->setState(new JumpingState());
        } else if (input == Input::S) {
            character->setState(new WalkingState()); // 停止跑步
        }
        // ... 其他输入处理
    }
    void update(Character* character, float deltaTime) override {
        // 跑步状态下的更新逻辑
    }
};

// 具体状态类:跳跃
class JumpingState : public CharacterState {
public:
    void handleInput(Character* character, Input input) override {
        //跳跃状态不允许其他输入
    }
    void update(Character* character, float deltaTime) override {
        // 跳跃状态下的更新逻辑,例如模拟重力
        // 在跳跃结束时切换到站立状态
        if (/* 跳跃结束条件 */) {
            character->setState(new StandingState());
        }
    }
};

// 角色类
class Character {
private:
    CharacterState* state;

public:
    Character(CharacterState* initialState) : state(initialState) {}

    void setState(CharacterState* newState) {
        delete state; // 避免内存泄漏
        state = newState;
    }

    void handleInput(Input input) {
        state->handleInput(this, input);
    }

    void update(float deltaTime) {
        state->update(this, deltaTime);
    }
};

enum class Input {
    W,
    S,
    A,
    D,
    Shift,
    Space
};

int main() {
    Character* character = new Character(new StandingState());
    character->handleInput(Input::W); // 进入行走状态
    character->update(0.1f); // 更新角色状态
    character->handleInput(Input::Shift); // 进入跑步状态
    character->update(0.1f); // 更新角色状态
    character->handleInput(Input::Space); // 进入跳跃状态
    character->update(0.1f); // 更新角色状态

    delete character;
    return 0;
}

在这个例子中,

Character
类是上下文,
CharacterState
是抽象状态类,
StandingState
WalkingState
RunningState
JumpingState
是具体状态类。通过状态模式,我们可以将角色的状态逻辑分离出来,使得代码更加模块化和易于维护。 实际游戏开发中,状态切换的条件会更复杂,可能需要考虑动画播放完毕、碰撞检测结果等因素。

如何在C++状态模式中处理状态的初始化和销毁?

状态的初始化通常在状态类被创建时进行,可以在构造函数中完成。 状态的销毁则需要在状态对象不再使用时进行,以避免内存泄漏。

在上面的例子中,

Context
类(
Character
类)负责管理状态对象的生命周期。当状态切换时,需要先
delete
当前状态对象,再创建新的状态对象。

void Context::setState(State* newState) {
    delete state; // 避免内存泄漏
    state = newState;
    state->handle(this);
}

//Character::setState类似

如果状态对象需要进行复杂的初始化或销毁操作,可以考虑使用工厂模式来创建状态对象,并在工厂类中处理状态的初始化和销毁逻辑。

另外,可以使用智能指针来管理状态对象的生命周期,例如

std::unique_ptr
std::shared_ptr
。 使用智能指针可以自动管理内存,避免手动
delete
状态对象,从而减少内存泄漏的风险。

#include 

class Context {
private:
    std::unique_ptr state;

public:
    Context(std::unique_ptr initialState) : state(std::move(initialState)) {}

    void setState(std::unique_ptr newState) {
        state = std::move(newState);
        state->handle(this);
    }

    void request() {
        state->handle(this);
    }
};

int main() {
    Context context(std::make_unique());
    context.request();
    context.request();

    return 0;
}

使用

std::unique_ptr
可以确保只有一个指针指向状态对象,当
Context
对象销毁时,状态对象也会自动销毁。 这种方式可以有效避免内存泄漏,并简化代码。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

15

2025.11.27

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1078

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

169

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1366

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

16

2026.01.19

数据库Delete用法
数据库Delete用法

数据库Delete用法:1、删除单条记录;2、删除多条记录;3、删除所有记录;4、删除特定条件的记录。更多关于数据库Delete的内容,大家可以访问下面的文章。

274

2023.11.13

drop和delete的区别
drop和delete的区别

drop和delete的区别:1、功能与用途;2、操作对象;3、可逆性;4、空间释放;5、执行速度与效率;6、与其他命令的交互;7、影响的持久性;8、语法和执行;9、触发器与约束;10、事务处理。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

213

2023.12.29

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

407

2023.08.14

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 7.7万人学习

C 教程
C 教程

共75课时 | 4.2万人学习

C++教程
C++教程

共115课时 | 14.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号