c++中如何实现简单的状态机_c++ switch与enum实现流程控制【汇总】
技术百科
尼克
发布时间:2026-01-19
浏览: 次 enum class + switch 是最轻量可控的状态机实现,核心是用枚举抽象离散互斥状态,循环中显式分支处理并更新状态,需覆盖所有枚举值、防 fall-through、封装复杂逻辑为函数。
用 enum 定义状态,switch 驱动流转
最轻量、最可控的状态机实现方式就是靠 enum + switch。核心是把每个状态抽象成一个命名常量,再在循环中根据当前状态做分支处理,并显式更新状态变量。
关键点在于:状态必须是离散、互斥、可穷举的;每次处理完一个状态后,必须明确赋值给下一个状态(哪怕维持原状),不能依赖隐式跳转。
-
enum class比裸enum更安全,避免命名污染和隐式转换 - 状态变量类型必须与
enum严格一致,比如State current = State::IDLE; -
switch必须覆盖所有枚举值,或加default:处理未定义状态(建议抛异常或断言)
enum class State { IDLE, RUNNING, PAUSED, STOPPED };
State current = State::IDLE;
while (running) {
switch (current) {
case State::IDLE:
if (start_requested()) current = State::RUNNING;
break;
case State::RUNNING:
do_work();
if (pause_requested()) current = State::PAUSED;
else if (stop_requested()) current = State::STOPPED;
break;
case State::PAUSED:
if
(resume_requested()) current = State::RUNNING;
break;
case State::STOPPED:
reset_resources();
current = State::IDLE;
break;
default:
throw std::runtime_error("invalid state: " + std::to_string(static_cast(current)));
}
}
避免 switch 中遗漏 break 导致状态“穿透”
这是实际编码中最常踩的坑:少写一个 break,程序会顺序执行后续 case 分支,造成状态逻辑错乱——比如本该停在 PAUSED,却意外触发了 STOPPED 的清理逻辑。
编译器不一定报错,尤其当多个 case 共享逻辑时,容易误以为是故意“fall-through”。C++17 起可用 [[fallthrough]] 显式标注,但绝大多数状态流转场景都不该穿透。
- 每个
case块末尾都应有break、return或显式状态变更 - 启用编译器警告:
-Wimplicit-fallthrough(GCC/Clang)能捕获潜在穿透 - 用
enum class+switch时,default:不仅防崩溃,还能暴露漏处理的状态
状态迁移逻辑别全塞进 switch,提取成独立函数
当某个状态的处理逻辑变长(比如要校验输入、调用多个子系统、涉及异步等待),硬塞在 case 里会让主循环臃肿难读,也破坏单一职责。
更合理的做法是把每个状态的“进入动作”“持续行为”“退出条件”封装成小函数,switch 只负责分发和状态跃迁。
- 函数命名体现意图,例如
on_enter_running()、update_paused()、should_transition_to_stopped() - 状态变更仍由主循环控制,不放在子函数内部(避免隐藏控制流)
- 若需跨状态共享数据,用结构体或类成员承载,而非全局变量
struct Machine {
State state = State::IDLE;
int progress = 0;
void step() {
switch (state) {
case State::IDLE: on_idle(); break;
case State::RUNNING: on_running(); break;
case State::PAUSED: on_paused(); break;
case State::STOPPED: on_stopped(); break;
}
}
private:
void on_running() {
progress++;
if (progress >= 100) state = State::STOPPED;
}
void on_stopped() {
progress = 0;
state = State::IDLE;
}
};
什么时候不该用 enum+switch?
这种写法适合状态数少(
- 状态之间存在复杂依赖或条件组合(比如 “只有在 A 且非 B 时才能从 X 到 Y”)→ 需要状态迁移表或策略模式
- 需要记录历史状态、支持回退或撤销 → 手动维护状态栈成本高,更适合用状态模式 + 虚函数
- 状态数动态变化,或由配置驱动 →
enum编译期固定,此时用字符串映射或整数 ID + 查表更灵活 - 多个对象共用同一组状态逻辑 → 重复写
switch易出错,应抽取为模板或基类
真正难的不是写对一个 switch,而是想清楚哪些状态必须显式建模、哪些边界条件必须拦截、哪些迁移应该禁止——这些设计决策不会被语法检查捕获,得靠反复推演和测试覆盖。
# 放在
# 这是
# 多个
# 还能
# 会让
# 什么时候
# mac
# default
# 循环
# c++
# 隐式转换
# class
# 编码
# 异步
# 封装
# 结构体
# 隐式
# switch
# break
# 穷举
# enum
# 全局变量
# 常量
# 变量类型
# 互斥
# 最轻
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- Python项目回滚策略_发布安全说明【指导】
- Mac如何解压zip和rar文件?(推荐免费工具)
- 如何在网页无标准表格标签时高效提取结构化数据
- 如何使用Golang配置安全开发环境_防止敏感信息
- 为什么本地php环境运行php脚本卡顿_php执行
- Python文件和流处理指南_高效读写大体积数据文
- 如何使用Golang读取日志文件_Golang b
- Win11怎么设置鼠标宏_Win11鼠标按键自定义
- Windows10如何重置此电脑_Windows1
- PHP主流架构怎么处理表单验证_规则与自定义【技巧
- Win11怎么开启游戏模式_Win11优化游戏帧数
- Windows 10怎么录屏_Windows 10
- Python迭代器生成器进阶教程_节省内存与懒加载
- Mac上的iMovie如何剪辑视频?(新手入门教程
- Windows服务持续崩溃怎样修复_系统服务保护机
- 如何在Golang中捕获HTTP服务器错误_Gol
- php订单日志怎么记录物流_php记录订单物流变更
- Laravel 查询 JSON 列:高效筛选包含数
- 如何将竖排文本文件转换为横排字符串
- Python对象比较与排序_魔术方法解析【教程】
- php怎么下载安装后测试是否成功_简单脚本验证方法
- php打包exe如何加密代码_防反编译保护方法【技
- 如何使用Golang benchmark测量函数延
- c++怎么编写动态链接库dll_c++ __dec
- Win11如何连接Xbox手柄 Win11蓝牙连接
- Win11怎么设置屏保_Windows 11屏幕保
- 如何在Golang中使用encoding/gob序
- Win11怎么关闭搜索历史 Win11清除搜索框最
- MAC怎么使用表情符号面板_MAC Emoji快捷
- MAC怎么解压RAR格式文件_MAC第三方解压工具
- PythonPandas数据分析教程_数据清洗与处
- Win11局域网共享怎么设置 Win11文件夹网络
- Windows10系统怎么查看CPU温度_Win1
- Win11怎么恢复误删照片_Win11数据恢复工具
- Win11怎么打开旧版计算器_Win11恢复传统计
- 如何用列表一次性对 DataFrame 的指定列应
- MySQL 中使用 IF 和 CASE 实现查询字
- Win11怎么退出高对比度模式_Win11取消反色
- Win11怎么关闭系统推荐内容_Windows11
- Windows 10怎么隐藏特定更新补丁_Wind
- Python与GPU加速技术_CUDA与Numba
- 获取 PHP 文件最后修改时间的正确方法
- php下载安装包怎么选_threadsafe与nt
- php485能和物联网模块通信吗_php485对接
- Win11怎么关闭定位服务_保护Win11位置隐私
- 如何解决同一段404代码在不同主机上表现不一致的问
- Win11怎么设置任务栏大小_Windows11注
- Linux如何挂载新硬盘_Linux磁盘分区格式化
- 如何自定义Windows终端的默认配置文件?(Po
- Go 中 defer 语句在 goroutine


QQ客服