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. BZOJ 3193: [JLOI2013]地形生成 计数 + 组合 + 动态规划

    第一问: 先不考虑山的高度有相同的:直接按照高度降序排序,试着将每一座山插入到前面山的缝隙中. 当然,这并不代表这些山的相对位置是固定的,因为后面高度更低的山是有机会插入进来的,所以就可以做到将所有情 ...

  2. C语言 - C语言简介

    一.C语言的发展史 C语言的发展离不开B语言,我们先从B语言说起.在1970年时,贝尔实验室的肯.汤姆逊将BCPL语言进行修改,由于肯.汤姆逊是从BCPL语言进行修改,因此,此时B语言就诞生了,B的含 ...

  3. rabbitmq 和 kafka 简单的性能测试

    测试环境:ubuntu 15.10 64位 cpu:inter core i7-4790 3.60GHZ * 8 内存:16GB 硬盘:ssd 120GB 软件环境:rabbmitmq 3.6.0   ...

  4. codevs 1057 津津的储蓄计划 2004年NOIP全国联赛提高组 x

     时间限制: 1 s  空间限制: 128000 KB   题目描述 Description 津津的零花钱一直都是自己管理.每个月的月初妈妈给津津300元钱,津津会预算这个月的花销,并且总能做到实际花 ...

  5. Spring Boot教程(二十)开发Web应用(1)

    静态资源访问 在我们开发Web应用的时候,需要引用大量的js.css.图片等静态资源. 默认配置 Spring Boot默认提供静态资源目录位置需置于classpath下,目录名需符合如下规则: /s ...

  6. sublime text3 - vue修改data,视图无更新

    ubuntu系统使用sublime text3做vue开发的时候遇到了一个问题,就是修改vue文件并保存后视图页面并不会随之修改,只有重新run dev时修改才会生效,原因没找到 猜想应该是subli ...

  7. sqli-labs(35)

    0X01 构造闭合 发现不需要闭合 ?id=- union ,database(), 0X02组合拳打法

  8. yum install ntp 报错:Error: Package: ntp-4.2.6p5-25.el7.centos.2.x86_64 (base)

    redhat7 在安装ntp时报如下错误 Error: Package: ntp-4.2.6p5-25.el7.centos.2.x86_64 (base) Requires: ntpdate = 4 ...

  9. Starting MySQL... ERROR! The server quit without updating PID file (/usr/local/mysql/data/VM_0_6_centos.pid)

    刚接触MySql数据库,参考一些文章后搭建起来了也创建了数据库,程序跑到很好,一觉醒来突然连接不上了 MySql数据库了. 研究了好一会才找到原因. 现象: 登录数据库失败 [root@VM_0_6_ ...

  10. 一致性hash算法Consistent Hashing

    一致性hash算法Consistent Hashing 对于原有hash算法hash%n so... 1.话不多说直接上代码,原理或详解自行百度即可 import cn.pheker.utils.Ut ...