状态机实践入门

2024-08-24 01:32
文章标签 实践 入门 状态机

本文主要是介绍状态机实践入门,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

状态机实践入门

原文转自:http://www.cnblogs.com/pang123hui/archive/2012/02/04/2338275.html 

不用怀疑,单片机的万能语言就是状态机。还希望大家不要条件反射式的看到状态机就 以为我要讲什么VHDL的东西——状态机是一种思维模式,是计算机理论的立足之本(不 相信请参考清华大学出版社的《自动机理论与应用》)——因此状态机的实现与语言本身关系并不是绝对的。本文要讨论的状态机,从实现方式上更类似于Java中常用的那种思维模式,而与VHDL相去甚远。 
  路要一步一步走,饭要一口一口吃,为了不把后来人吓跑,状态机理论中更多复杂的部分,我会在以后专门写文章讨论,这里我先找一个切入点,从我常用的2种状态机编写方式为大家慢慢展开。 


  首先,关于几个问题,比如:什么地方用状态机?状态机究竟有几种写法?状态机效率 到底高不高?是不是把简单问题弄复杂了?这类问题统统不在本文讨论之列,简而言之——谁用谁知道。其实,还不能简单的就这么下了结论,套八股文而不求甚解的也大有人在————因此我要说:关于状态机的各种问题“谁思考谁实践谁坚持谁知道”。


状态机入门第一式:switch case一线到底 
要点:    用switch结构配合一个状态变量,通过修改状态变量的值来切换状态。 
范例:

复制代码
 //! 定义状态名称与状态值之间的关系,增加可读性 
#define FSM_START 0x00
#define FSM_STATE_A 0x01
#define FSM_STATE_B 0x02

#define FSM_RESET 0xFF
bool fsm_example_A( <形参列表> ) {
static uint8_t s_chFSMState = FSM_START; //!< 定义状态变量

switch ( s_chFSMState ) {
case FSM_START:
//! 这里添加状态机初始化代码

s_chFSMState = FSM_STATE_A; //!< 进入下一状态
break;
case FSM_STATE_A:
//! 这里添加状态机A进入下一状态的检测代码
if (<某某条件>) {
//! 这里做一些进入下一状态时要做的准备工作
s_chFSMState = FSM_STATE_B; //!< 进入下一状态
}
break;
case FSM_STATE_B:
//! 这里添加状态机A进入下一状态的检测代码
if (<某某条件>) {
//! 这里做一些进入下一状态时要做的准备工作
s_chFSMState = FSM_STATE_A; //!< 进入下一状态
} else if (<某某条件>) {
} else if (<某某条件>) {

} else {
}
break;

case FSM_STOP:
case FSM_RESET:
default:
//! 这里添加状态机复位相关的代码

chFSMState = FSM_START; //!< 状态机复位

//! 返回false表示状态机已经不需要继续运行了
return false;
}

//! 返回true表示状态机正在运行
return true;
}
复制代码

总结:    从范例可知,这种状态机就是一根筋……并不是说他走不出什么分支来,而是说通常他没有办法让多个状态同时处于激活状态。

 

状态机入门第二式:if 判断变化无穷 
要点:     用if else…else if结构的组合来描述状态流程图。 
什么是状态流程图?我不想多解释,因为就那么个简单的东西,说多了反而神秘兮兮的。状态流程图你可以简单粗暴的认为,他就是流程图,等你用得多了,你就渐渐明白为啥多了“状态”二字;如果你后来或者先前学过状态图,那么很快你就会明白状态流程图比状态图“高级”了多少。 
1、 不管怎么说,你可以先为你要处理的事物画一个流程图。如果流程图都不会画,就 
    不用凑热闹了。 
2、 接下来,把流程图上每一个方框或者判断筐都“简单粗暴”地看成一个状态。 
3、 将每一个状态用if结构表示出来 

if (<状态标志>) { 
//! 状态代码

}

4、 自己看着办,合并多余的状态,优化优化代码。 
范例:   

复制代码
//! 首先将布尔量的状态标志压缩在一个字节里面以节省内存开支 
typedef union {
uint8_t Value;
uint8_t Byte;
struct {
unsigned BIT0:1;
unsigned BIT1:1;
unsigned BIT2:1;
unsigned BIT3:1;
unsigned BIT4:1;
unsigned BIT5:1;
unsigned BIT6:1;
unsigned BIT7:1;
} Bits;
}byte_t;

#define FSM_ACTION_FLAG s_tbState.Bits
#define FSM_STOP_ALL_ACTIONS() do {s_tbState.Value = 0;}while(0)
#define FSM_START (0 == s_tbState.Value)
#define FSM_STATE_A FSM_ACTION_FLAG.BIT0
#define FSM_STATE_B FSM_ACTION_FLAG.BIT1

#define FSM_STATE_H FSM_ACTION_FLAG.BIT7

bool fsm_example_B( <形参列表> ) {
static byte_t s_tbState = {0}; //!< 定义状态变量

if (FSM_START) { //!< 起始状态
//! 这里放置状态机初始化的代码

FSM_STATE_A = true; //!< 进入状态B,start装台自动结束
}

if (FSM_STATE_A) { //!< 一个典型的简单状态
//! 这里放置状态A的代码或者

//! 这里放置某些条件以开启别的状态
if (<某些条件>) {
//! 这里做一些“进入”下一个状态之前的准备工作
FSM_STATE_B = true; //!< 开启下一个状态
FSM_STATE_A = false; //!< 结束当前状态
}
}

if (FSM_STATE_B) { //!< 一个典型的监视状态

//! 这里检测某些条件
if (<某些条件>) {
//! 这里做一些“开启”某个状态的准备工作
FSM_STATE_C = true; //!< 开启某一个状态而不结束当前状态
FSM_STATE_D = true; //!< 你当然可以一次触发多个状态

} else if (<某些条件>) {
//! 满足某些条件以后关闭当前状态
FSM_STATE_B = false;
}
}

if (FSM_STATE_F) { //!< 一个典型的子状态机调用
if (!fsm_example_a(<实参列表>)) { //!< 等待子状态机返回false
//!子状态机运行完成,进入下一状态

FSM_STATE_F = false; //!< 结束当前状态
FSM_STATE_x = true; //!< 进入下一状态x代表某个字母
}
}

if (FSM_STATE_H) { //!< 一个典型的中止状态
//!< 某些状态机的操作,比如释放某些资源

FSM_STOP_ALL_ACTIONS(); //!< 复位状态机
return false; //!< 返回false表示状态机结束
}

return true; //!< 返回true表示状态机保持运行
}
复制代码

总结:    从范例可知,这种状态机非常灵活,通过布尔变量的开启和关闭,你可以自由的控制某些状态的开启。同一时刻可能有多个状态是激活的。这种结构几乎可以翻译任何流程图。具体还有很多好处,可以在使用中体会。 


状态机入门第三式:状态在心中,无态也变态 
要点:    所有的函数都可以看作是状态机,只不过普通的函数是一个只有单一状态的状态机。如果函数有返回值,且这个返回值能表征至少两种以上不同的状态(比如返回是一个指针,那么NULL和非NULL就是两种状态;比如返回是一个布尔变量,那么true和false就是两种状态;比如返回的是一个整数,并且整数的某些特征可以被分类,那么这些不同分类就是几种不同的状态),那么这些返回值就可以被用作指示当前状态机的运行情况。状态机可以调用子状态机。所有的状态都应该是none-block的,简单说就是不会把系统定死在某一个状态里面很久都出不来,比如while(1)或者循环次数较大的for结构;否则状态机的存在意义就大打折扣——直接按照流程图写代码不就好了,干吗非要翻译成状态机?状态机中,状态的功能应该是等待某一个事件的发生(或者说条件的满足);某些情况下,一些一次性执行完的流程也可以独立成一个状态——它当然没有等待任何条件的满足,你可以认为他是无条件进行状态转移的。 


总体说来,状态机是一个万能的计算机语言表述方式,与具体的载体语言关系不大。心中有状态,代码怎会无状态?状态机是裸机条件下多任务的廉价实现方案。在状态机多任务条件下,操作系统牵涉的几乎所有概念都会有所涉及,比如任务的同步,临界区的保护,任务间的通讯,任务的优先级,资源的动态分配等等。你可以这么理解,每一个状态机都是一个进程,每一个状态都是一个线程,因为进程有自己的资源,而同一个进程里面的多个线程是公用同一片资源的。你甚至可以在有抢占式操作系统的情况下用状态机,这个时候,操作系统的每一个任务都是个内核,那么整个系统开发就可以登小于 
多核系统开发了。是不是很有意思?没意思就别看了!状态机的种种,以后再表。 


Have a good time. 

这篇关于状态机实践入门的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

在Java中使用OpenCV实践

《在Java中使用OpenCV实践》用户分享了在Java项目中集成OpenCV4.10.0的实践经验,涵盖库简介、Windows安装、依赖配置及灰度图测试,强调其在图像处理领域的多功能性,并计划后续探... 目录前言一 、OpenCV1.简介2.下载与安装3.目录说明二、在Java项目中使用三 、测试1.测

MyBatis-Plus 自动赋值实体字段最佳实践指南

《MyBatis-Plus自动赋值实体字段最佳实践指南》MyBatis-Plus通过@TableField注解与填充策略,实现时间戳、用户信息、逻辑删除等字段的自动填充,减少手动赋值,提升开发效率与... 目录1. MyBATis-Plus 自动赋值概述1.1 适用场景1.2 自动填充的原理1.3 填充策略

Olingo分析和实践之EDM 辅助序列化器详解(最佳实践)

《Olingo分析和实践之EDM辅助序列化器详解(最佳实践)》EDM辅助序列化器是ApacheOlingoOData框架中无需完整EDM模型的智能序列化工具,通过运行时类型推断实现灵活数据转换,适用... 目录概念与定义什么是 EDM 辅助序列化器?核心概念设计目标核心特点1. EDM 信息可选2. 智能类

Olingo分析和实践之OData框架核心组件初始化(关键步骤)

《Olingo分析和实践之OData框架核心组件初始化(关键步骤)》ODataSpringBootService通过初始化OData实例和服务元数据,构建框架核心能力与数据模型结构,实现序列化、URI... 目录概述第一步:OData实例创建1.1 OData.newInstance() 详细分析1.1.1

Spring Boot 与微服务入门实战详细总结

《SpringBoot与微服务入门实战详细总结》本文讲解SpringBoot框架的核心特性如快速构建、自动配置、零XML与微服务架构的定义、演进及优缺点,涵盖开发环境准备和HelloWorld实战... 目录一、Spring Boot 核心概述二、微服务架构详解1. 微服务的定义与演进2. 微服务的优缺点三

Olingo分析和实践之ODataImpl详细分析(重要方法详解)

《Olingo分析和实践之ODataImpl详细分析(重要方法详解)》ODataImpl.java是ApacheOlingoOData框架的核心工厂类,负责创建序列化器、反序列化器和处理器等组件,... 目录概述主要职责类结构与继承关系核心功能分析1. 序列化器管理2. 反序列化器管理3. 处理器管理重要方

从入门到精通详解LangChain加载HTML内容的全攻略

《从入门到精通详解LangChain加载HTML内容的全攻略》这篇文章主要为大家详细介绍了如何用LangChain优雅地处理HTML内容,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录引言:当大语言模型遇见html一、HTML加载器为什么需要专门的HTML加载器核心加载器对比表二

虚拟机Centos7安装MySQL数据库实践

《虚拟机Centos7安装MySQL数据库实践》用户分享在虚拟机安装MySQL的全过程及常见问题解决方案,包括处理GPG密钥、修改密码策略、配置远程访问权限及防火墙设置,最终通过关闭防火墙和停止Net... 目录安装mysql数据库下载wget命令下载MySQL安装包安装MySQL安装MySQL服务安装完成

从入门到进阶讲解Python自动化Playwright实战指南

《从入门到进阶讲解Python自动化Playwright实战指南》Playwright是针对Python语言的纯自动化工具,它可以通过单个API自动执行Chromium,Firefox和WebKit... 目录Playwright 简介核心优势安装步骤观点与案例结合Playwright 核心功能从零开始学习

SpringBoot整合(ES)ElasticSearch7.8实践

《SpringBoot整合(ES)ElasticSearch7.8实践》本文详细介绍了SpringBoot整合ElasticSearch7.8的教程,涵盖依赖添加、客户端初始化、索引创建与获取、批量插... 目录SpringBoot整合ElasticSearch7.8添加依赖初始化创建SpringBoot项