golang 状态机
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 状态机的更多相关文章
- golang开源项目qor快速搭建网站qor-example运行实践
最近想找几个基于Go语言开发的简单的开源项目学习下,分享给大家,github上有心人的收集的awesome-go项目集锦:github地址 发现一个Qor项目: Qor 是基于 Golang 开发的的 ...
- 消息/事件, 同步/异步/协程, 并发/并行 协程与状态机 ——从python asyncio引发的集中学习
我比较笨,只看用await asyncio.sleep(x)实现的例子,看再多,也还是不会. 已经在unity3d里用过coroutine了,也知道是“你执行一下,主动让出权限:我执行一下,主动让出权 ...
- Golang后台开发初体验
转自:http://blog.csdn.net/cszhouwei/article/details/37740277 补充反馈 slice 既然聊到slice,就不得不提它的近亲array,这里不太想 ...
- Golang优秀开源项目汇总, 10大流行Go语言开源项目, golang 开源项目全集(golang/go/wiki/Projects), GitHub上优秀的Go开源项目
Golang优秀开源项目汇总(持续更新...)我把这个汇总放在github上了, 后面更新也会在github上更新. https://github.com/hackstoic/golang-open- ...
- 面试必问:Golang高阶-Golang协程实现原理
引言 实现并发编程有进程,线程,IO多路复用的方式.(并发和并行我们这里不区分,如果CPU是多核的,可能在多个核同时进行,我们叫并行,如果是单核,需要排队切换,我们叫并发) 进程和线程的区别 进程是计 ...
- 数据结构和算法(Golang实现)(6)简单入门Golang-并发、协程和信道
并发.协程和信道 Golang语言提供了go关键字,以及名为chan的数据类型,以及一些标准库的并发锁等,我们将会简单介绍一下并发的一些概念,然后学习这些Golang特征知识. 一.并发介绍 我们写程 ...
- 找工作面试题记录与参考资料(Golang/C++/计算机网络/操作系统/算法等)
记录下去年(2020年)找工作的面试题及参考资料. C++ 智能指针的实现原理 多态的实现原理[2] C++11/14/17新特性[3] 手写memcpy和memmove[4] 介绍下boost库 计 ...
- Golang, 以17个简短代码片段,切底弄懂 channel 基础
(原创出处为本博客:http://www.cnblogs.com/linguanh/) 前序: 因为打算自己搞个基于Golang的IM服务器,所以复习了下之前一直没怎么使用的协程.管道等高并发编程知识 ...
- 说说Golang的使用心得
13年上半年接触了Golang,对Golang十分喜爱.现在是2015年,离春节还有几天,从开始学习到现在的一年半时间里,前前后后也用Golang写了些代码,其中包括业余时间的,也有产品项目中的.一直 ...
随机推荐
- js for循环 框架内部的
var head001 =true; var head002 = true; var head003 = true; ; h++) { console.log(h); } h=; ;h < ; ...
- layui 源码解读(部分)
<!DOCTYPE html> <head> </head> <body> <input type="button" id=& ...
- promql 常用函数介绍
Metrics类型 根据不同监控指标之间的差异,Prometheus定义了4中不同的指标类型(metric type):Counter(计数器).Gauge(仪表盘).Histogram(直方图).S ...
- rollup的学习
概述(Overview) Rollup 是一个 JavaScript 模块打包器,可以将小块代码编译成大块复杂的代码,例如 library 或应用程序.Rollup 对代码模块使用新的标准化格式,这些 ...
- HBase调优案例(三)——Spark访问HBase慢
负载信息:RegionServer:3个 Region:5400多个 现象:在使用Spark对HBase进行scan操作时发现有些task执行比较慢 原因分析:查看Spark应用的executor日志 ...
- 使用SSH和Nginx做内网HTTP映射
https://blog.csdn.net/imdyf/article/details/80143991
- 一、Robotframework安装步骤
1.安装python并验证安装成功 a.安装python-2.7.14.amd64------默认路径安装即可 b.添加环境变量path:C:\Python27; C:\Python27\Script ...
- 3、Shiro授权
Shiro授权过程和认证过程相似: 项目结构: package com.shiro.shiroframe; import org.apache.shiro.SecurityUtils; import ...
- Tomcat使用介绍
一.tomcat介绍 Tomcat服务器是一个免费的开放源代码的轻量级Web 应用服务器,如apache处理静态HTML能力突出不同,tomcat处理动态HTML能力相当强大,因此一般项目都是部署ap ...
- CSS3——对齐 组合选择符 伪类 伪元素 导航栏 下拉菜单
水平&垂直对齐 元素居中对齐 .center { margin: auto; width: 50%; border: 3px solid green; padding: 10px; } 文本 ...