Go有限状态机实现和实战

news2024/12/14 22:47:08

Go有限状态机实现和实战

在这里插入图片描述

有限状态机

什么是状态机

有限状态机(Finite State Machine, FSM)是一种用于建模系统行为的计算模型,它包含有限数量的状态,并通过事件或条件实现状态之间的转换。FSM的状态数量是有限的,因此称为有限状态机

关于有限的解释:也就是被描述的事物的状态的数量是有限的,例如开关的状态只有“开”和“关”两个;灯的状态只有“亮”和“灭”等等。

特点

状态机的状态数量是有限的,在确定的状态下,给定特定的事件,系统会转换到明确的下一个状态。

作用

使用状态机来表达状态的流转,会使语义会更加清晰,会增强代码的稳定性可控性可维护性

适用场景

面对复杂的状态流转都可以使用状态机来实现

状态机是描述系统行为的强大工具,其核心是状态、事件和状态转换。它的清晰逻辑和确定性使其在复杂流程控制、嵌入式设备、游戏开发等领域非常适用,为系统开发提供了简洁、高效的解决方案。

状态机核心概念

状态机有四个核心概念,这是所有状态机的基础

  • State:状态。一个状态机至少要包含两个状态。
  • Transition:过渡。也就是从一个状态变化为另一个状态(即状态转移)。
  • Event:事件。事件会触发状态转移,也就是状态转移的条件
  • Action:动作。事件发生以后要执行动作,即事件处理

有限状态机的工作原理如图所示,发生事件(event)后,根据当前状态(cur_state) ,决定执行的动作(action),并设置下一个状态(next_state)。

Go状态机设计

我们根据上面的概念设计一个状态机

示例:自动贩卖机状态机

在这里插入图片描述

package main

import (
	"fmt"
)

// 定义状态类型
type State string
//  1.State:状态。一个状态机至少要包含两个状态。
const (
	StateIdle     State = "Idle(空闲)"     // 空闲状态
	StateCoin     State = "Coin(投币)"     // 投币状态
	StateItem     State = "Item(选择商品)"     // 选择商品状态
)

// 定义事件类型
type Event string
// 2.Event:事件。事件会触发状态转移,也就是状态转移的条件。
const (
	EventInsertCoin  Event = "InsertCoin"  // 投币事件
	EventSelectItem  Event = "SelectItem"  // 选择商品事件
	EventDispense    Event = "Dispense"    // 出货事件
)

// 定义状态转移 
// 3.Transition:过渡。也就是从一个状态变化为另一个状态(即状态转移)。
const(
    Transitions = map[State]map[Event]State{
			StateIdle: {
				EventInsertCoin: StateCoin,
			},
			StateCoin: {
				EventSelectItem: StateItem,
			},
			StateItem: {
				EventDispense: StateIdle,
			},
    }
)


// 定义状态机结构体
type FSM struct {
	currentState State
	transitions  map[State]map[Event]State // 状态-事件转换映射
}
// 创建状态机
func NewFSM() *FSM {
	return &FSM{
		currentState: StateIdle, // 初始状态
        transitions: Transitions,
	}
}

// 处理事件并进行状态转换
// 4.Action:动作。事件发生以后要执行动作,即事件处理。
func (fsm *FSM) HandleEvent(event Event) {
	nextState, ok := fsm.transitions[fsm.currentState][event]
	if !ok {
		fmt.Printf("事件 %s 无法从状态 %s 转换\n", event, fsm.currentState)
		return
	}
	fmt.Printf("从状态 %s 转到状态 %s 通过事件 %s\n", fsm.currentState, nextState, event)
	fsm.currentState = nextState
}

// 获取当前状态
func (fsm *FSM) CurrentState() State {
	return fsm.currentState
}

func main() {
	fsm := NewFSM()

	// 测试状态机
	fsm.HandleEvent(EventInsertCoin)  // 投币,进入选择商品状态
	fsm.HandleEvent(EventSelectItem)  // 选择商品,准备出货
	fsm.HandleEvent(EventDispense)    // 出货,返回初始状态
}

运行结果

从状态 Idle(空闲) 转到状态 Coin(投币) 通过事件 InsertCoin(投币事件)
从状态 Coin(投币) 转到状态 Item(选择商品) 通过事件 SelectItem(选择商品事件)
从状态 Item(选择商品) 转到状态 Idle(空闲) 通过事件 Dispense(出货事件)

代码解释:

  • State(对应State) :定义了有限状态机的状态和事件。状态有 Idle、Coin、Item 和 Dispense。

  • Event(对应Event):事件有 InsertCoin、SelectItem、Dispense 和 ReturnCoins。

  • Transitions(对应Transition):定义了状态转移规则,即当某个状态收到某个事件时,会转移到另一个状态。

  • HandleEvent(对应Action):该方法处理事件,并根据当前状态和事件查找下一个状态。如果找不到对应的转换,则表示事件在当前状态下无效。

  • FSM 结构体:负责存在状态机的信息,FSM 包含当前状态和状态转换规则。transitions 映射定义了每个状态在不同事件下的下一个状态。

事件处理

我们可以将Action进行拓展,对状态机进行扩展,比如添加日志记录、超时处理、错误处理、通知等。
在这里插入图片描述
具体实现我们可以参考下面生产使用的代码。

生产使用

上面的代码是一个简单的状态机,实际使用中,需要考虑更多的因素,比如状态的初始化、状态的持久化、状态的恢复、状态的监听等。

  1. 状态的初始化:在创建状态机时,需要初始化当前状态,并设置初始状态的转换规则。

  2. 状态的持久化:在状态机中,需要将状态的转换记录持久化到数据库或者文件中,以便在系统重启时恢复状态。

  3. 状态的恢复:在系统重启时,需要根据持久化的状态记录恢复状态机。

  4. 状态的监听:在状态机中,需要监听状态的变化,以便在状态发生变化时进行通知。

  5. 状态的校验:在状态机中,需要校验状态的转换是否合法,比如不允许从空闲状态直接进入选择商品状态。

  6. 状态的错误处理:在状态机中,需要处理状态转换过程中出现的错误,比如转换失败、超时等。

  7. 状态的日志记录:在状态机中,需要记录状态的转换记录,以便后续分析

生产上我们可以使用 github.com/looplab/fsm
这个库提供了一些常用的功能,如状态的初始化、状态的持久化、状态的恢复、状态的监听等。

package fsm_demo

import (
	"context"
	"fmt"
	"testing"

	"github.com/looplab/fsm"
)

// 定义状态类型
type State string

// 1.State:状态。一个状态机至少要包含两个状态。
const (
	StateIdle State = "Idle(空闲)"   // 空闲状态
	StateCoin State = "Coin(投币)"   // 投币状态
	StateItem State = "Item(选择商品)" // 选择商品状态
)

// 定义事件类型
type Event string

// 2. Event:事件。事件会触发状态转移,也就是状态转移的条件。
const (
	EventInsertCoin Event = "InsertCoin" // 投币事件
	EventSelectItem Event = "SelectItem" // 选择商品事件
	EventDispense   Event = "Dispense"   // 出货事件
)

type FSM struct {
	Id  string   // 数据id
	FSM *fsm.FSM // 状态机
}

func NewFSM(id string) *FSM {
	d := &FSM{
		Id: id,
	}

	d.FSM = fsm.NewFSM(
		string(StateIdle),
        // 3. Transition 定义转换
		fsm.Events{
			{Name: string(EventInsertCoin), Src: []string{string(StateIdle)}, Dst: string(StateCoin)},
			{Name: string(EventSelectItem), Src: []string{string(StateCoin)}, Dst: string(StateItem)},
			{Name: string(EventDispense), Src: []string{string(StateItem)}, Dst: string(StateIdle)},
		},
        // 4. Action 定义动作
		fsm.Callbacks{
			"before_event": func(ctx context.Context, e *fsm.Event) {
				fmt.Println(e.Args[0])
				e.FSM.SetMetadata("error", "error")
				fmt.Println("before_event 通用(可以记录日志)", e.Event, e.FSM.Current())
				e.Err = fmt.Errorf("error")
				e.Cancel()
				// e.Event = "error"
			},
			fmt.Sprintf("before_%s", EventInsertCoin): func(ctx context.Context, e *fsm.Event) {
				fmt.Println("before", EventInsertCoin, e.Event, e.FSM.Current())
				// e.Err = fmt.Errorf("error")
			},
			fmt.Sprintf("enter_%s", StateCoin): func(ctx context.Context, e *fsm.Event) {
				fmt.Println("enter", StateCoin, e.Event, e.FSM.Current())
				// e.Err = fmt.Errorf("error")
			},
			"enter_state": func(_ context.Context, e *fsm.Event) { d.enterState(e) },
			fmt.Sprintf("after_%s", EventInsertCoin): func(ctx context.Context, e *fsm.Event) {
				fmt.Println("after", EventInsertCoin, e.Event, e.FSM.Current())
				// e.Err = fmt.Errorf("error")
			},
			"after_event": func(_ context.Context, e *fsm.Event) {
				fmt.Println(e.FSM.Metadata("error"))
				fmt.Println("after_event 通用(可以记录日志)", e.Event, e.FSM.Current())
			},
		},
	)
	return d
}

// 更新状态
func (d *FSM) enterState(e *fsm.Event) {
	// called after entering all states
	// todo 更新数据库,持久化
	fmt.Println("enter_state 通用 更新数据库", e.FSM.Current())
}

func Test_Main(t *testing.T) {
	plan := NewFSM(string(StateIdle))

	err := plan.FSM.Event(context.Background(), string(EventInsertCoin), 1)
	if err != nil {
		fmt.Println(err)
	}
	fmt.Println(plan.FSM.Current())
	err = plan.FSM.Event(context.Background(), string(EventSelectItem))
	if err != nil {
		fmt.Println(err)
	}
	fmt.Println(plan.FSM.Current())
	err = plan.FSM.Event(context.Background(), string(EventDispense))
	if err != nil {
		fmt.Println(err)
	}
	fmt.Println(plan.FSM.Current())
	err = plan.FSM.Event(context.Background(), string(EventDispense))
	if err != nil {
		fmt.Println(err)
	}
	fmt.Println(plan.FSM.Current())
}

使用去起来和我们自己写的代码类似,也是定义 State ,Evnent,Transition,Action。这个是所有FSM通用的。

这个库可以很灵活的配置事件处理。可以配置很多前置和后置事件。

// 1. before_<EVENT> - 在名为 <EVENT> 的事件之前调用
// 2. before_event - 在所有事件之前调用
// 3. leave_<OLD_STATE> - 在离开 <OLD_STATE> 之前调用
// 4. leave_state - 在离开所有状态之前调用
// 5. enter_<NEW_STATE> - 在进入 <NEW_STATE> 之后调用
// 6. enter_state - 在进入所有状态之后调用
// 7. after_<EVENT> - 在名为 <EVENT> 的事件之后调用
// 8. after_event - 在所有事件之后调用

而且错误处理机制也很完善。使用这个库后大家可以把更多的精力关注业务层。

参考:

https://github.com/looplab/fsm

https://www.cnblogs.com/huageyiyangdewo/p/17351310.html

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

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

相关文章

Qt实现自定义行编辑器

引言 开发环境项目结构ui界面设计示例代码运行效果总结qt中原有的行编辑器无法满足当前的需要,所以需要自定义行编辑器。 通过上下按键切换到不同的行编辑器,在选中的行编辑器中输入数字,编辑器呈现边框,编辑后按下回车键保存之前编辑的数值,没有按下回车键直接切换上下键…

企业级日志分析系统ELK之ELK概述

ELK 概述 ELK 介绍 什么是 ELK 早期IT架构中的系统和应用的日志分散在不同的主机和文件&#xff0c;如果应用出现问题&#xff0c;开发和运维人员想排 查原因&#xff0c;就要先找到相应的主机上的日志文件再进行查找和分析&#xff0c;所以非常不方便&#xff0c;而且还涉及…

2024153读书笔记|《春烂漫:新平摄影作品选》——跳绳酷似人生路,起落平常,进退平常,莫惧征途万里长

2024153读书笔记|《春烂漫&#xff1a;新平摄影作品选》——跳绳酷似人生路&#xff0c;起落平常&#xff0c;进退平常&#xff0c;莫惧征途万里长 《春烂漫&#xff1a;新平摄影作品选》作者新平&#xff0c;2019.12.25年读完的小书&#xff0c;当时就觉得挺不错&#xff0c;今…

每日一站技術架構解析之-cc手機桌布網

# 網站技術架構解析&#xff1a; ## 一、整體架構概述https://tw.ccwallpaper.com是一個提供手機壁紙、桌布免費下載的網站&#xff0c;其技術架構設計旨在實現高效的圖片資源管理與用戶訪問體驗優化。 ### &#xff08;一&#xff09;前端展示 1. **HTML/CSS/JavaScript基礎構…

Nacos 3.0 Alpha 发布,在安全、泛用、云原生更进一步

自 2021 年发布以来&#xff0c;Nacos 2.0 在社区的支持下已走过近三年&#xff0c;期间取得了诸多成就。在高性能与易扩展性方面&#xff0c;Nacos 2.0 取得了显著进展&#xff0c;同时在易用性和安全性上也不断提升。想了解更多详细信息&#xff0c;欢迎阅读我们之前发布的回…

Mybatis -plus -jion的复习

Mybatis -plus -jion 是一个 MyBatis-Plus 的增强工具&#xff0c;在 MyBatis-Plus 的基础上只做增强不做改变&#xff0c;为简化开发、提高效率而生。 - **无侵入**&#xff1a;只做增强不做改变&#xff0c;引入它不会对现有工程产生影响&#xff0c;如丝般顺滑 - **损耗小…

计算机网络-Wireshark探索ARP

使用工具 Wiresharkarp: To inspect and clear the cache used by the ARP protocol on your computer.curl(MacOS)ifconfig(MacOS or Linux): to inspect the state of your computer’s network interface.route/netstat: To inspect the routes used by your computer.Brows…

Qt之自定义动态调控是否显示日志

创作灵感 最近在芯驰x9hp上开发仪表应用。由于需要仪表警告音&#xff0c;所以在该平台上折腾并且调试仪表声音的时候&#xff0c;无意间发现使用&#xff1a; export QT_DEBUG_PLUGINS1 可以打印更详细的调试信息。于是想着自己开发的应用也可以这样搞&#xff0c;这样更方便…

右玉200MW光伏电站项目 微气象、安全警卫、视频监控系统

一、项目名称 山西右玉200MW光伏电站项目 微气象、安全警卫、视频监控系统 二、项目背景&#xff1a; 山西右玉光伏发电项目位于右玉县境内&#xff0c;总装机容量为200MW&#xff0c;即太阳能电池阵列共由200个1MW多晶硅电池阵列子方阵组成&#xff0c;每个子方阵包含太阳能…

Java基础学习:java常用启动命令

一、java -jar 1、系统属性传递 使用形式&#xff1a;java -DpathD:\jacoco -jar 获取方式&#xff1a;System.getProperties() 2、系统参数传递 使用形式&#xff1a;java -jar application.jar --jacocoPathD:\tomcat 获取方式&#xff1a;通过启动方法入口main的参数arg…

一级考试真题(2019)

一级考试真题&#xff08;2019&#xff09;

echarts图表自定义配置(二)——代码封装

下图是初版&#xff0c;火山图的代码。可以看出&#xff0c;里面的变量&#xff0c;逻辑&#xff0c;函数存在冗余&#xff0c;基本上都是改了参数&#xff0c;同样的get和set&#xff0c;去刷新图表&#xff1b;对于往后继续开发十几二十个图表&#xff0c;会很麻烦。因此需要…

如何用状态图进行设计05

到目前为止&#xff0c;我们已经讨论了状态图的原理。这些原理对状态图和扩展状态图都适用。第二章后面的部分主要讲述了扩展状态图的扩展功能。我们将围绕这些增强的功能&#xff0c;使你对BetterState Pro的设计能力有很好的了解。 关于这些内容和其他有关扩展状态图特性的完…

鸿蒙NEXT开发案例:九宫格随机

【引言】 在鸿蒙NEXT开发中&#xff0c;九宫格抽奖是一个常见且有趣的应用场景。通过九宫格抽奖&#xff0c;用户可以随机获得不同奖品&#xff0c;增加互动性和趣味性。本文将介绍如何使用鸿蒙开发框架实现九宫格抽奖功能&#xff0c;并通过代码解析展示实现细节。 【环境准…

【efinance一个2k星的库】

efinance 是一个可以快速获取基金、股票、债券、期货数据的 Python 库&#xff0c;回测以及量化交易的好帮手 但没有等比复权的&#xff0c;不用。 import efinance as ef ef.stock.get_quote_history(510880,fqt2)

协同办公软件新升级:细节优化,让办公更简单

细节决定成败&#xff0c;企业酷信协同办公系统通过贴近客户实际需求的一系列改进和创新&#xff0c;在技术架构、系统结构、管理理念和使用性能上&#xff0c;都达到了国内先进水平&#xff0c;同时具备独特的优势。让我们看看企业酷信是如何通过这些细节提升&#xff0c;为企…

【AI知识】有监督学习分类任务之支持向量机

1.支持向量机概念 支持向量机&#xff08;Support Vector Machine, SVM&#xff09; 是一种有监督学习算法&#xff0c;主要用于分类任务&#xff08;也可用于回归任务&#xff0c;即支持向量回归&#xff0c;SVR&#xff09;。SVM的核心思想是找到一个最优的超平面&#xff0…

MySQL有哪些高可用方案?

大家好&#xff0c;我是锋哥。今天分享关于【MySQL有哪些高可用方案?】面试题。希望对大家有帮助&#xff1b; MySQL有哪些高可用方案? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 MySQL 高可用方案旨在确保数据库系统的高可靠性、低宕机时间、以及在硬件故障…

利用Docker分层构建优化镜像大小

合适docker镜像文件大小不仅影响容器启动效率&#xff0c;也影响资源占用效率。本文介绍如何利用分层方式构建docker镜像&#xff0c;采用多种方式避免镜像文件太大而影响性能。 Docker 镜像大小优化的重要性 资源利用效率 较小的镜像文件在存储和传输过程中占用更少的空间和带…

threejs——无人机概念切割效果

主要技术采用着色器的切割渲染,和之前写的风车可视化的文章不同,这次的切割效果是在着色器的基础上实现的,并新增了很多可调节的变量,兄弟们,走曲儿~ 线上演示地址,点击体验 源码下载地址,点击下载 正文 从图中大概可以看出以下信息,一个由线组成的无人机模型,一个由…