状态机quick中是一个亮点,假设我们做一款RPG游戏,一个角色通常会拥有idle,attack,walk。run,death这些状态,假设游戏角色的状态採用分支条件推断的话。会造成很庞大而难以维护。但一旦使用了状态机这样的模式。就会显得简单方便。

对于quick中的状态机是怎样实现的咱们先不去了解。首先看看怎样去使用它。

总结起来,假设让一个类拥有状态机,主要有两步:

1.创建状态机对象

2.初始化状态机,主要包含事件和回调函数

1.创建状态机组件

self.fsm = {}
cc.GameObject.extend(self.fsm):addComponent("components.behavior.StateMachine"):exportMethods()

这样就创建了一个状态机对象,接下来我们要对其初始化,事实上也就是设置各个状态的逻辑。

2.初始化状态机(设置状态逻辑)

设置状态逻辑是重写setupState方法,这当中有这么几个字段參数,

  • initial:状态机的初始状态
  • terminal (final):结束状态
  • events:状态发生转变时相应的事件
  • callbacks:发生转变时的回调函数

一般我们会设置initial,events和callbacks这三个。

先看events,在events中须要分清楚“事件”和“状态”,events採用table结构,比如我们来写一个

events = {
{name = "move", from = {"idle", "jump"}, to = "walk"},

这当中move是事件,就像触摸事件event.name那样,name表示事件名称,而from和to后面跟的idle。jump。walk表示状态。所以上面的意思就是。当运行move事件时,假设状态是idle或者jump。那么都会跳转到walk状态上。

from的状态能够是单一状态,也能够使集合状态。就是几个状态,但to的状态仅仅能唯一,不然程序还给你来个随机状态?肯定不行的。

所以这里须要想好我们的主角有哪些状态。当什么事件发生时。他会从什么状态变到什么状态上去。比如我简单这么写。

events = {
{name = "move", from = {"idle", "jump"}, to = "walk"},
{name = "attack", from = {"idle", "walk"}, to = "jump"},
{name = "normal", from = {"walk", "jump"}, to = "idle"},
},

解释一下,假设是normal事件。无论主角在走路walk还是跳跃jump,都会变成闲置idle状态。其它同理。

接下来一个重点是callbacks參数,

即所谓回调了,就是事件触发,会运行一系列的函数。

  • onbeforeEVNET: 在事件EVENT開始前被激活
  • onleaveSTATE: 在离开旧状态STATE时被激活
  • onenterSTATE 或 onSTATE:在进入新状态STATE时被激活
  • onafterEVENT 或 onEVENT:在事件EVENT结束后被激活

比如

callbacks = {
onenteridle = function () --或者 onidle
print("idle")
end,
},

此外还有5种通用型的回调来捕获全部事件和状态的变化:

  • onbeforeevent: 在不论什么事件開始前被激活
  • onleavestate: 在离开不论什么状态时被激活
  • onenterstate:在进入不论什么状态时被激活
  • onafterevent :在不论什么事件结束后被激活
  • onchangestate :当状态发生改变的时候被激活
这里面的名称是不能够改动的,它是针对于不论什么事件和不论什么状态的。
所以大家能够想象一下这当中有多少事件回调和多少状态回调,它们的先后顺序。咱们能够自己分别print一下就知道调用的先后了。这里就不演示了。

最后,就是调用这些事件了,通过self.fsm:doEvent(event)就能够了,參数event相应events參数名称。

此外还有这些,

  • fsm:isReady() :返回状态机是否就绪
  • fsm:getState() :返回当前状态
  • fsm:isState(state) :推断当前状态是否是參数state状态
  • fsm:canDoEvent(eventName) :当前状态假设能完毕eventName相应的event的状态转换,则返回true
  • fsm:cannotDoEvent(eventName) :当前状态假设不能完毕eventName相应的event的状态转换,则返回true
  • fsm:isFinishedState() :当前状态假设是终于状态,则返回true
  • fsm:doEventForce(name, ...) :强制对当前状态进行转换

接下来在实际运用一下,我们创建一个Player类,为其加入一个状态机,
local Player = class("Player", function ()
return display.newSprite("icon.png")
end) function Player:ctor()
self:addStateMachine()
end function Player:doEvent(event)
self.fsm:doEvent(event)
end function Player:addStateMachine()
self.fsm = {}
cc.GameObject.extend(self.fsm):addComponent("components.behavior.StateMachine"):exportMethods() self.fsm:setupState({
initial = "idle", events = {
{name = "move", from = {"idle", "jump"}, to = "walk"},
{name = "attack", from = {"idle", "walk"}, to = "jump"},
{name = "normal", from = {"walk", "jump"}, to = "idle"},
}, callbacks = {
onenteridle = function ()
local scale = CCScaleBy:create(0.2, 1.2)
self:runAction(CCRepeat:create(transition.sequence({scale, scale:reverse()}), 2))
end, onenterwalk = function ()
local move = CCMoveBy:create(0.2, ccp(100, 0))
self:runAction(CCRepeat:create(transition.sequence({move, move:reverse()}), 2))
end, onenterjump = function ()
local jump = CCJumpBy:create(0.5, ccp(0, 0), 100, 2)
self:runAction(jump)
end,
},
})
end return Player
比較简单,回调函数仅仅是写了进入三个状态的回调,然后为Player加入一个doEvent函数,调用状态机中doEvent。

回到我们的MyScene.lua中,
local Player = import("..views.Player")

local MyScene = class("MyScene", function ()
return display.newScene("MyScene")
end) function MyScene:ctor() local player = Player.new()
player:setPosition(display.cx, display.cy)
self:addChild(player) local function menuCallback(tag)
if tag == 1 then
player:doEvent("normal")
elseif tag == 2 then
player:doEvent("move")
elseif tag == 3 then
player:doEvent("attack")
end
end local mormalItem = ui.newTTFLabelMenuItem({text = "normal", x = display.width*0.3, y = display.height*0.2, listener = menuCallback, tag = 1})
local moveItem = ui.newTTFLabelMenuItem({text = "move", x = display.width*0.5, y = display.height*0.2, listener = menuCallback, tag = 2})
local attackItem = ui.newTTFLabelMenuItem({text = "attack", x = display.width*0.7, y = display.height*0.2, listener = menuCallback, tag = 3})
local menu = ui.newMenu({mormalItem, moveItem, attackItem})
self:addChild(menu) end return MyScene

加入我们刚才的Player。记得import或者require,这里为了方便我就通过菜单button的形式来分别doEvent了。

点击打开链接



Quick StateMachine状态机的更多相关文章

  1. Spring Boot 揭秘与实战(七) 实用技术篇 - StateMachine 状态机机制

    文章目录 1. 环境依赖 2. 状态和事件 2.1. 状态枚举 2.2. 事件枚举 3. 状态机配置4. 状态监听器 3.1. 初始化状态机状态 3.2. 初始化状态迁移事件 5. 总结 6. 源代码 ...

  2. Spring Boot - StateMachine状态机

    是Spring Boot提供的状态机的现成实现. 理论(有点像工作流) 需要定义一些状态的枚举,以及一些引起状态变化的事件的枚举. 每个状态可以对应的创建一个继承自org.springframewor ...

  3. 使用Spring StateMachine框架实现状态机

    spring statemachine刚出来不久,但是对于一些企业的大型应用的使用还是十分有借鉴意义的. 最近使用了下这个,感觉还是挺好的. 下面举个例子来说下吧: 创建一个Spring Boot的基 ...

  4. 管理订单状态,该上状态机吗?轻量级状态机COLA StateMachine保姆级入门教程

    前言 在平常的后端项目开发中,状态机模式的使用其实没有大家想象中那么常见,笔者之前由于不在电商领域工作,很少在业务代码中用状态机来管理各种状态,一般都是手动get/set状态值.去年笔者进入了电商领域 ...

  5. mina statemachine解读(一)

      statemachine(状态机)在维护多状态数据时有非常好的作用,现在github上star排名最前的是squirrel-foundation以及spring-statemachine,而min ...

  6. 状态机引擎在vivo营销自动化中的深度实践 | 引擎篇02

    本文是<vivo营销自动化技术解密>的第3篇文章,分析了营销自动化业务背景和状态机引入原因.状态机的三种基本实现方式,同时介绍了几种业界流行的开源状态机框架实现和特点,以及在项目开发过程中 ...

  7. bk.

    http://ol.tgbus.com/zt2013/gzsnew/ 巴士盘点 十大游戏工作室 http://bbs.3dmgame.com/forum.php?mod=viewthread& ...

  8. jsplumb 中文教程

    https://wdd.js.org/jsplumb-chinese-tutorial/#/ 1. jsplumb 中文基础教程 后续更新会在仓库:https://github.com/wangdua ...

  9. C#复习笔记(5)--C#5:简化的异步编程(异步编程的深入分析)

    首先,阐明一下标题的这个“深入分析”起得很惭愧,但是又不知道该起什么名字,这个系列也主要是做一些复习的笔记,供自己以后查阅,如果能够帮助到别人,那自然是再好不过了. 然后,我想说的是异步方法的状态机真 ...

随机推荐

  1. windows socket----select模型

    一般我们的网络编程都是用bind ,listen,accept,send/sendto,recv/recvfrom.在创建套接字的时候,是默认使用阻塞模式的,每当我们调用send/sendto等方法时 ...

  2. ExtJs4 笔记(9) Ext.Panel 面板控件、 Ext.window.Window 窗口控件、 Ext.container.Viewport 布局控件

    本篇讲解三个容器类控件. 一.面板控件 Ext.Panel 一个面板控件包括几个部分,有标题栏.工具栏.正文.按钮区.标题栏位于最上面,工具栏可以在四个位置放置,围绕中间部分正文,按钮区位于最小方.下 ...

  3. 推动Common Lisp的实际应用

    推动Common Lisp的实际应用 推动Common Lisp的实际应用

  4. cape town

    开普敦_百度百科 开普敦

  5. IP分类地址——a,b,c 类是如何划分的

    今天IP网络使用32位地址,点分十进制格式,如172.16.0.0.地址格式:IP地址=网络地址+主机地址 或 IP地址=主机地址+子网地址+主机地址. IP地址类型 当互联网最初的设计,为了便于网络 ...

  6. 虚拟机centOS中安装Redis,主机Redis Destop Manager不能访问虚拟机Redis server的解决方案

    今天在学些redis的时候碰到个问题,发现主机Redis Destop Manager不能访问虚拟机Redis server的解决方案,找了一些网上的资料,原因可能有两个,整理记录下来: 1. Red ...

  7. ruby on rails创建的页面訪问很慢

    ruby on rails创建的页面訪问很慢 用rvm安装的ruby1.9.3 解决:cd ~/.rvm/rubies/ruby-1.9.3-p547/lib/ruby/1.9.1/webrick v ...

  8. hdu 4291 A Short problem(矩阵+取模循环节)

    A Short problem                                                          Time Limit: 2000/1000 MS (J ...

  9. Java Web----Java Web的数据库操作(二)

    Java Web的数据库操作 三.JDBC操作数据库 上一篇介绍了JDBC API,之后就可以通过API来操作数据库,实现对数据库的CRUD操作了. http://blog.csdn.net/zhai ...

  10. mysql导出和导入数据库

    出口 在dos计划,切换到mysql按照该文件夹bin下一个.输入以下命令 mysqldump -u root -p nxu_life > nxu_life2.sql 运行完毕后,就能够看到在b ...