package main

import (
"errors"
"fmt"
"reflect"
) type State interface {
Name() string
EnableSameTransit() bool
OnBegin()
OnEnd()
CanTransitTo(name string) bool
} func StateName(s State) string {
if s == nil{
return "none"
}
return reflect.TypeOf(s).Elem().Name()
} type StateInfo struct {
name string
} func (s *StateInfo) Name() string {
return s.name
}
func (s *StateInfo) setName(name string){
s.name = name
}
//是否能够同状态转移
func (s *StateInfo) EnableSameTransit()bool{
return false
}
func (s *StateInfo) OnBegin(){ }
func (s *StateInfo) OnEnd(){ }
//是否能够转移到状态
func (s *StateInfo) CanTransitTo(name string) bool{
return true
} type StateManager struct {
stateByName map[string]State
Onchange func(from,to State)
curr State
} func (sm *StateManager) Add(s State){
name := StateName(s)
s.(interface{
setName(name string)
}).setName(name)
if sm.Get(name)!=nil{
panic("重复添加:"+name)
}
sm.stateByName[name] = s
} func (sm *StateManager) Get(name string) State {
if v,ok:=sm.stateByName[name];ok{
return v
}
return nil
}
func NewStateManager()*StateManager{
return &StateManager{
stateByName:make(map[string]State),
}
}
var ErrForbidSameTransit = errors.New("ErrForbidSameTransit")
var ErrCannotTransitToState = errors.New("ErrCannotTransitToState")
var ErrStateNotFound = errors.New("ErrStateNotFound") func (sm *StateManager)CurrState()State {
return sm.curr
} func (sm *StateManager) CanCurrTransitTo(name string)bool {
if sm.curr == nil{
return true
}
if sm.curr.Name() == name && !sm.curr.EnableSameTransit(){
return true
}
return sm.curr.CanTransitTo(name)
}
func (sm *StateManager) Transit(name string)error{
next := sm.Get(name)
if next==nil{
return ErrStateNotFound
}
pre := sm.curr
if sm.curr!=nil{
if sm.curr.Name() == name && !sm.curr.EnableSameTransit(){
return ErrForbidSameTransit
}
if !sm.CanCurrTransitTo(name){
return ErrCannotTransitToState
}
sm.curr.OnEnd()
}
sm.curr = next
sm.curr.OnBegin()
if sm.Onchange!=nil{
sm.Onchange(pre,sm.curr)
}
return nil
} type IdleState struct {
StateInfo
} func (i *IdleState)OnBegin() {
fmt.Println("idle on begin")
}
func (i *IdleState)OnEnd() {
fmt.Println("idle on end")
} type MoveState struct {
StateInfo
} func (i *MoveState)OnBegin() {
fmt.Println("move on begin")
}
func (i *MoveState)OnEnd() {
fmt.Println("move on end")
} func (i *MoveState)EnableSameTransit() bool {
return true
} type JumpState struct {
StateInfo
} func (i *JumpState)OnBegin() {
fmt.Println("jump on begin")
}
func (i *JumpState)OnEnd() {
fmt.Println("jump on end")
}
func (i *JumpState) CanTransitTo(name string) bool {
return name != "MoveState"
} func main() {
sm := NewStateManager()
sm.Onchange = func(from, to State) {
fmt.Println("from",from,"to",to)
}
sm.Add(new(IdleState))
sm.Add(new(JumpState))
sm.Add(new(MoveState))
sm.Transit("JumpState")
sm.Transit("IdleState")
sm.Transit("MoveState")
}

  

golang 状态机的更多相关文章

  1. golang开源项目qor快速搭建网站qor-example运行实践

    最近想找几个基于Go语言开发的简单的开源项目学习下,分享给大家,github上有心人的收集的awesome-go项目集锦:github地址 发现一个Qor项目: Qor 是基于 Golang 开发的的 ...

  2. 消息/事件, 同步/异步/协程, 并发/并行 协程与状态机 ——从python asyncio引发的集中学习

    我比较笨,只看用await asyncio.sleep(x)实现的例子,看再多,也还是不会. 已经在unity3d里用过coroutine了,也知道是“你执行一下,主动让出权限:我执行一下,主动让出权 ...

  3. Golang后台开发初体验

    转自:http://blog.csdn.net/cszhouwei/article/details/37740277 补充反馈 slice 既然聊到slice,就不得不提它的近亲array,这里不太想 ...

  4. Golang优秀开源项目汇总, 10大流行Go语言开源项目, golang 开源项目全集(golang/go/wiki/Projects), GitHub上优秀的Go开源项目

    Golang优秀开源项目汇总(持续更新...)我把这个汇总放在github上了, 后面更新也会在github上更新. https://github.com/hackstoic/golang-open- ...

  5. 面试必问:Golang高阶-Golang协程实现原理

    引言 实现并发编程有进程,线程,IO多路复用的方式.(并发和并行我们这里不区分,如果CPU是多核的,可能在多个核同时进行,我们叫并行,如果是单核,需要排队切换,我们叫并发) 进程和线程的区别 进程是计 ...

  6. 数据结构和算法(Golang实现)(6)简单入门Golang-并发、协程和信道

    并发.协程和信道 Golang语言提供了go关键字,以及名为chan的数据类型,以及一些标准库的并发锁等,我们将会简单介绍一下并发的一些概念,然后学习这些Golang特征知识. 一.并发介绍 我们写程 ...

  7. 找工作面试题记录与参考资料(Golang/C++/计算机网络/操作系统/算法等)

    记录下去年(2020年)找工作的面试题及参考资料. C++ 智能指针的实现原理 多态的实现原理[2] C++11/14/17新特性[3] 手写memcpy和memmove[4] 介绍下boost库 计 ...

  8. Golang, 以17个简短代码片段,切底弄懂 channel 基础

    (原创出处为本博客:http://www.cnblogs.com/linguanh/) 前序: 因为打算自己搞个基于Golang的IM服务器,所以复习了下之前一直没怎么使用的协程.管道等高并发编程知识 ...

  9. 说说Golang的使用心得

    13年上半年接触了Golang,对Golang十分喜爱.现在是2015年,离春节还有几天,从开始学习到现在的一年半时间里,前前后后也用Golang写了些代码,其中包括业余时间的,也有产品项目中的.一直 ...

随机推荐

  1. PHP基础教程 10款人气暴涨的PHP开源工具

    若想创建动态而又新颖的Web应用程序,PHP便是理想的选择.不用说,在Web开发世界里,PHP是最流行的语言之一.一些非常好用的PHP开源工具着实拯救了不少开发任务繁重的PHP开发 人员,减轻他们的开 ...

  2. [模板] Kruskal算法 && 克鲁斯卡尔重构树

    克鲁斯卡尔重构树 发现没把板子放上来... 现在放一下 克鲁斯卡尔算法的正确性是利用反证法证明的. 简要地说, 就是如果不加入当前权值最小的边 \(e_1\), 那么之后加入的边和这条边会形成一个环. ...

  3. 洛谷 P4151 BZOJ 2115 [WC2011]最大XOR和路径

    //bzoj上的题面太丑了,导致VJ的题面也很丑,于是这题用洛谷的题面 题面描述 XOR(异或)是一种二元逻辑运算,其运算结果当且仅当两个输入的布尔值不相等时才为真,否则为假. XOR 运算的真值表如 ...

  4. 1.Python编程基础

    1. 其实,程序指的就是一系列指令,用来告诉计算机做什么,而编写程序的关键在于,我们需要用计算机可以理解的语言来提供这些指令. 虽然借助 Siri(Apple).Google Now(Android) ...

  5. intellij idea中去除@Autowired注入对象的红色波浪线提示

    idea中通过@Autowired注入的对象一直有下划线提示. 解决:改变@Autowired的检查级别即可. 快捷键:Ctrl+Alt+s,进入idea设置界面,输入inspections检索

  6. 百度编辑器ueditor上传图片失败,显示上传错误,实际上图片已经传到服务器或者本地

    报错,上传失败,图片没有显示,且调试response没有信息,但是图片已经上传到了本地 这个问题是因为ueditor里面的Upload.class.php里面__construct()方法里面的ico ...

  7. always_populate_raw_post_data

    Deprecated: Automatically populating $HTTP_RAW_POST_DATA is deprecated and will be removed in a futu ...

  8. C# Setting.settings . 用法

    1.定义 在Settings.settings文件中定义配置字段.把作用范围定义为:User则运行时可更改,Applicatiion则运行时不可更改.可以使用数据网格视图,很方便: 2.读取配置值 t ...

  9. Collector的使用

    一.Collector的引入 1)Collector的聚合作用前面已经使用过,将list.stream后的一系列操作之后再返回list. 2)Collector的引入,通过需求:将绿色的Apple放在 ...

  10. 一:flask-第一个flask程序

    安装flask:pip install flask,或者pycharm安装 最小模型 访问 后台: