C++ 状态模式详解

状态模式(State Pattern)是一种行为设计模式,它允许一个对象在内部状态改变时改变其行为,使对象看起来像是改变了其类

核心概念

设计原则

状态模式遵循以下设计原则:

  1. 单一职责原则:将状态相关行为分离到不同类中

  2. 开闭原则:可以新增状态而不修改现有代码

  3. 封装性:状态转换逻辑封装在状态类中

主要优点

  1. 清晰状态转换:将状态转换逻辑组织化

  2. 减少条件语句:消除庞大的条件状态判断

  3. 可扩展性:容易添加新状态和转换

  4. 行为局部化:状态特定行为集中在对应状态类中

模式结构

主要组件

  1. Context(上下文)

    • 维护当前状态的引用

    • 将状态相关请求委托给当前状态对象

  2. State(状态接口)

    • 定义状态接口

    • 封装与Context特定状态相关的行为

  3. ConcreteState(具体状态)

    • 实现状态接口

    • 每个子类实现与Context状态相关的行为

完整代码示例

#include <iostream>
#include <memory>
#include <string>// 前置声明
class State;// ==================== 上下文类 ====================
class Player {std::unique_ptr<State> state_;std::string name_;public:explicit Player(const std::string& name);void setState(std::unique_ptr<State> state);std::string getName() const;// 玩家行为void play();void pause();void stop();void lock();void unlock();
};// ==================== 状态接口 ====================
class State {
public:virtual ~State() = default;virtual void play(Player* player) = 0;virtual void pause(Player* player) = 0;virtual void stop(Player* player) = 0;virtual void lock(Player* player) = 0;virtual void unlock(Player* player) = 0;protected:void changeState(Player* player, std::unique_ptr<State> state) {player->setState(std::move(state));}
};// ==================== 具体状态 ====================
class ReadyState : public State {
public:void play(Player* player) override;void pause(Player* player) override { /* 无效操作 */ }void stop(Player* player) override { /* 无效操作 */ }void lock(Player* player) override;void unlock(Player* player) override { /* 已在解锁状态 */ }
};class PlayingState : public State {
public:void play(Player* player) override { /* 已在播放状态 */ }void pause(Player* player) override;void stop(Player* player) override;void lock(Player* player) override;void unlock(Player* player) override { /* 无效操作 */ }
};class PausedState : public State {
public:void play(Player* player) override;void pause(Player* player) override { /* 已在暂停状态 */ }void stop(Player* player) override;void lock(Player* player) override;void unlock(Player* player) override { /* 无效操作 */ }
};class LockedState : public State {
public:void play(Player* player) override { /* 锁定状态下不能播放 */ }void pause(Player* player) override { /* 锁定状态下不能暂停 */ }void stop(Player* player) override { /* 锁定状态下不能停止 */ }void lock(Player* player) override { /* 已在锁定状态 */ }void unlock(Player* player) override;
};// ==================== 上下文方法实现 ====================
Player::Player(const std::string& name) : name_(name) {state_ = std::make_unique<ReadyState>();
}void Player::setState(std::unique_ptr<State> state) {state_ = std::move(state);
}std::string Player::getName() const {return name_;
}void Player::play() {state_->play(this);
}void Player::pause() {state_->pause(this);
}void Player::stop() {state_->stop(this);
}void Player::lock() {state_->lock(this);
}void Player::unlock() {state_->unlock(this);
}// ==================== 具体状态方法实现 ====================
void ReadyState::play(Player* player) {std::cout << player->getName() << ": 开始播放" << std::endl;changeState(player, std::make_unique<PlayingState>());
}void ReadyState::lock(Player* player) {std::cout << player->getName() << ": 锁定播放器" << std::endl;changeState(player, std::make_unique<LockedState>());
}void PlayingState::pause(Player* player) {std::cout << player->getName() << ": 暂停播放" << std::endl;changeState(player, std::make_unique<PausedState>());
}void PlayingState::stop(Player* player) {std::cout << player->getName() << ": 停止播放" << std::endl;changeState(player, std::make_unique<ReadyState>());
}void PlayingState::lock(Player* player) {std::cout << player->getName() << ": 锁定播放器" << std::endl;changeState(player, std::make_unique<LockedState>());
}void PausedState::play(Player* player) {std::cout << player->getName() << ": 恢复播放" << std::endl;changeState(player, std::make_unique<PlayingState>());
}void PausedState::stop(Player* player) {std::cout << player->getName() << ": 停止播放" << std::endl;changeState(player, std::make_unique<ReadyState>());
}void PausedState::lock(Player* player) {std::cout << player->getName() << ": 锁定播放器" << std::endl;changeState(player, std::make_unique<LockedState>());
}void LockedState::unlock(Player* player) {std::cout << player->getName() << ": 解锁播放器" << std::endl;changeState(player, std::make_unique<ReadyState>());
}// ==================== 客户端代码 ====================
int main() {std::cout << "=== 状态模式演示: 音乐播放器 ===" << std::endl;Player player("我的播放器");// 正常流程player.play();player.pause();player.play();player.stop();// 锁定测试std::cout << "\n测试锁定功能:" << std::endl;player.play();player.lock();player.play();  // 应该无效player.unlock();player.play();// 无效操作测试std::cout << "\n测试无效操作:" << std::endl;player.stop();player.stop();  // 应该无效player.pause(); // 应该无效return 0;
}

模式变体

1. 状态表驱动

class StateMachine {std::unordered_map<std::string, std::unordered_map<std::string, std::function<void()>>> transitions_;std::string currentState_;public:void addTransition(const std::string& from, const std::string& event, const std::string& to, std::function<void()> action) {transitions_[from][event] = [this, to, action] {action();currentState_ = to;};}void handleEvent(const std::string& event) {auto& stateTransitions = transitions_[currentState_];if (stateTransitions.find(event) != stateTransitions.end()) {stateTransitions[event]();}}
};

2. 共享状态对象

class SharedState : public State {// 可以共享的状态数据
};class ConcreteStateA : public SharedState {// 实现特定行为
};// 使用时可以共享同一个状态实例
auto sharedState = std::make_shared<ConcreteStateA>();
context1.setState(sharedState);
context2.setState(sharedState);

实际应用场景

  1. 工作流引擎:如订单状态流转

  2. 游戏开发:角色行为状态管理

  3. UI系统:控件不同状态下的行为

  4. 网络协议:协议状态机实现

  5. 硬件控制:设备状态管理

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/diannao/82740.shtml
繁体地址,请注明出处:http://hk.pswp.cn/diannao/82740.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Html5新特性_js 给元素自定义属性_json 详解_浅克隆与深克隆

文章目录 1. html5新特性2.用 js 给元素自定义属性3.json3.1 json与普通对象的区别3.2 json对象与 js对象的转化 4.浅克隆和深克隆 1. html5新特性 html5中引入了新的特性&#xff08;新的标签&#xff09;&#xff0c;下面的新标签是新的结构标签&#xff0c;不过不太常用 h…

std::move 和 std::forward

关联点 都是执行转换(cast)的函数&#xff08;函数模板&#xff09;&#xff0c;不产生任何可执行代码。且都可以把实参转换成右值。 std::move无条件将实参&#xff08;const除外 &#xff09;转换成右值引用&#xff0c;std::forward 条件返回右值引用 _EXPORT_STD template…

Uniapp编写微信小程序,使用canvas进行绘图

一、canvas文档&#xff1a; https://developer.mozilla.org/zh-CN/docs/Web/API/Canvas_API/Tutorial 二、数据绘制&#xff08;单位是像素&#xff09;&#xff1a; 1、绘制文本&#xff1a; 文字的长度超过设置的最大宽度&#xff0c;文字会缩在一起 ① 填充文本&#xf…

FLASH闪存(擦除、编译)

FLASH闪存 文章目录 FLASH闪存1.存储器映像位置2.FLASH简介3.闪存模块组织3.2闪存的共性&#xff1a; 4.FLASH基本结构4.1FLASH解锁4.2使用指针访问寄存器 5.选项字节5.1选项字节编程5.2选项字节擦除 6.相关函数介绍7.读取内部FLASH&#xff08;实操&#xff09;7.1接线图7.2工…

PostgreSQL 序列(Sequence) 与 Oracle 序列对比

PostgreSQL 序列(Sequence) 与 Oracle 序列对比 PostgreSQL 和 Oracle 都提供了序列(Sequence)功能&#xff0c;但在实现细节和使用方式上存在一些重要差异。以下是两者的详细对比&#xff1a; 一 基本语法对比 1.1 创建序列 PostgreSQL: CREATE [ { TEMPORARY | TEMP } |…

12.2.2 allocator类

allocator类将分配内存空间、调用构造函数、调用析构函数、释放内存空间这4部分操作分开&#xff0c;全部交给程序员来执行&#xff0c;不像new和delete #include <iostream> #include <string>int main() {const int n 10;std::allocator<std::string> al…

Android 中 Handler (创建时)内存泄漏问题及解决方案

一、Handler 内存泄漏核心原理 真题 1&#xff1a;分析 Handler 内存泄漏场景 题目描述&#xff1a; 在 Activity 中使用非静态内部类 Handler 发送延迟消息&#xff0c;旋转屏幕后 Activity 无法释放&#xff0c;分析原因并给出解决方案。 内存泄漏链路分析&#xff1a; 引…

SSTI记录

SSTI(Server-Side Template Injection&#xff0c;服务器段模板注入) 当前使用的一些框架&#xff0c;如python的flask、php的tp、java的spring&#xff0c;都采用成熟的MVC模式&#xff0c;用户的输入会先进入到Controller控制器&#xff0c;然后根据请求的类型和请求的指令发…

探索边缘计算:赋能物联网的未来

摘要 随着物联网&#xff08;IoT&#xff09;技术的飞速发展&#xff0c;越来越多的设备接入网络&#xff0c;产生了海量的数据。传统的云计算模式在处理这些数据时面临着延迟高、带宽不足等问题&#xff0c;而边缘计算的出现为解决这些问题提供了新的思路。本文将深入探讨边缘…

tabs切换#

1、html <el-tabs v-model"tabValue" tab-change"handleTabClick"><el-tab-pane label"集群" name"1"></el-tab-pane><el-tab-pane label"节点" name"2"></el-tab-pane></el-ta…

JSON 实体属性映射的最佳实践

一、结构与命名规范 ‌保持字段命名一致性‌ JSON 字段名与实体属性名应遵循统一的命名规则&#xff08;如驼峰命名或下划线分隔&#xff09;&#xff0c;避免因大小写差异导致映射失败。 // 使用 JsonProperty 显式指定映射关系&#xff08;Jackson&#xff09; public class …

hiveserver2与beeline进行远程连接hive配置及遇到的问题

1、hiveserver2 参与用户模拟功能&#xff0c;因为开启后才能保证各用户之间的权限隔离。 1.1、配置 $HADOOP_HOME/etc/hadoop/core-site.xml <!--配置所有节点的root用户都可作为代理用户--> <property><name>hadoop.proxyuser.root.hosts</name>&…

硅基计划2.0 学习总结 壹 Java初阶

一、初见Java &#xff08;1&#xff09;Java简介 首先不得不承认Java是一门优秀的程序设计语言 其系列的计算机软件和跨平台体系包括国内的生态链完善是C/C语言难以弥补的 &#xff08;2&#xff09;Java SE 全称Java Standard Edition&#xff0c;是Java体系的基础 &am…

nRF5_SDK_17.1.0_ddde560之ble_app_uart_c 出错

Error #541: ARM::CMSIS:CORE:5.3.0 component is missing (previously found in pack ARM.CMSIS.5.6.0) Error #541: NordicSemiconductor::Device:Startup:8.40.3 component is missing (previously found in pack NordicSemiconductor.nRF_DeviceFamilyPack.8.40.3) 下载n…

基于大模型预测的多发性硬化综合诊疗方案研究报告大纲

目录 一、引言二、文献综述三、大模型预测系统构建四、术前预测与手术方案制定五、术中监测与决策支持六、术后护理与并发症预测七、麻醉方案智能优化八、统计分析与技术验证九、实验验证与证据支持十、健康教育与指导系统十一、结论与展望一、引言 (一)研究背景与意义 多发…

bootstrap自助(抽样)法

一&#xff0c;概念 一言以蔽之&#xff1a;从训练集中有放回的均匀抽样——》本质就是有放回抽样&#xff1b; 自助法&#xff08;bootstrap&#xff09;是一种通过从数据集中重复抽样来估计统计量分布的非参数方法。它可用于构建假设检验&#xff0c;当对参数模型的假设存在…

用1W字讲透数据预处理,数据增强

大家好&#xff01;我是我不是小upper~ 今天咱们来聊聊数据增强 —— 这个在机器学习领域堪称 “数据魔法” 的实用技术&#xff01; 在深度学习的世界里&#xff0c;数据就像模型的 “养分”。数据的质量和数量&#xff0c;直接决定了模型最终能达到的 “高度”。当数据不足时…

无人机空中物流优化:用 Python 打造高效配送模型

友友们好! 我是Echo_Wish,我的的新专栏《Python进阶》以及《Python!实战!》正式启动啦!这是专为那些渴望提升Python技能的朋友们量身打造的专栏,无论你是已经有一定基础的开发者,还是希望深入挖掘Python潜力的爱好者,这里都将是你不可错过的宝藏。 在这个专栏中,你将会…

C++核心编程解析:模板、容器与异常处理全指南

文章目录 一、模板1.1 定义1.2 作用1.3 函数模版1.3.1 格式 1.4 类模版1.4.1 格式1.4.2 代码示例1.4.3 特性 二、容器2.1 概念2.2 容器特性2.3 分类2.4 向量vector2.4.1 特性2.4.2 初始化与操作2.4.3 插入删除 2.5 迭代器2.6 列表&#xff08;list&#xff09;2.6.1 遍历方式2.…

JWT的介绍与在Fastapi框架中的应用

什么是JWT JWT (JSON Web Token) 是一个开放标准 ( RFC 7519 )&#xff0c;它定义了一种紧凑且自包含的方式&#xff0c;用于在各方之间安全地以 JSON 对象的形式传输信息。由于这些信息经过数字签名&#xff0c;因此可以被验证和信任。JWT 可以使用密钥&#xff08;采用HMAC算…