【Unity/HFSM】使用UnityHFSM实现输入缓冲(预输入)和打断机制

文章目录

    • 前言
    • 预输入
      • Animancer的InputBuffer:
      • 在UnityHFSM中实现InputBuffer:
    • 打断机制

前言

参考Animancer在状态机中的InputBuffer,在UnityHFSM中实现类似的InputBuffer机制,同时扩展一个状态打断机制

插件介绍:

Animancer:Unity的动画插件,易于修改和扩展。

UnityHFSM:Github上的一个开源分层状态机项目


基于继承的方法使用UnityHFSM:

UnityHFSM有非常方便的使用方法,可以在不创建新类的情况下创建状态和状态机,定义转换条件等,但如果状态逻辑比较复杂,并且有统一的父级行为,则建议使用类继承的方式定义状态和状态机类。

本文是笔者在做一个自己的ARPG小项目时做出来的,思路仅供参考。


预输入

Animancer的InputBuffer:

Animancer的InputBuffer主要处理流程如下:

  1. ​ 在构造函数中绑定一个状态机
  2. ​ 外部调用Buffer函数,传入期望的目标状态和超时时长
  3. ​ InputBuffer在Update函数中轮询是否能够转换到目标状态,一旦转换成功即立刻结束轮询
  4. ​ 超时后也会结束轮询

注意点:

  1. ​ Animancer接收的是状态机接口,而不是具体的状态机类。但由于需要继承来扩展UntiyHFSM的状态机和状态行为的需要,我在实现时会传入的是类而非接口。
  2. ​ Animancer的InputBuffer并未继承自Monobehavior,并且没有使用任何UnityEngine的东西,即Update需要在其他Mono类中调用,Time.deltaTime也要靠外部传入,方便起见,我实现时直接使用Time.deltaTime。

在UnityHFSM中实现InputBuffer:

Animancer的状态机有TryResetState函数,并且会返回是否成功,UntiyHFSM中无此函数,故使用StateMachine类的StateChanged来触发事件,并与期望的目标状态进行比对,一致则终止轮询。

Animancer直接使用TryResetState函数来改变状态,状态是否允许进入的条件也写在状态中,UnityHFSM依赖于Transition定义两个指定状态之间的转换,虽然也有RequestStateChange来强制转换到某个状态,但这样做并不优雅,并且可能出错。由于这个操作的多样性和复杂性,由外界传入Action来决定轮询时该进行的操作。(大多数时候使用UnityHFSM的StateMachine的Trigger函数来转换状态)

以下是代码,CharacterStateMachineBase是我扩展的角色状态机基类,ECharacterState是角色状态的枚举值

public class InputBuffer
{public bool IsActive => Action != null;public float TimeOut;public Action Action;public CharacterStateMachineBase StateMachine;public ECharacterState TargetState;public InputBuffer(CharacterStateMachineBase stateMachine){StateMachine = stateMachine;StateMachine.StateChanged += OnStateChanged;}~InputBuffer(){StateMachine.StateChanged -= OnStateChanged;}public void Buffer(Action action, ECharacterState targetState, float timeOut){Action = action;TargetState = targetState;TimeOut = timeOut;}public void Update(){if (IsActive){Action();TimeOut -= Time.deltaTime;if (TimeOut < 0)Clear();}}public virtual void Clear(){Action = null;TimeOut = default;}public void OnStateChanged(UnityHFSM.StateBase<ECharacterState> state){if (IsActive && state.name == TargetState){Clear();}}
}

在外部调用时如下

//CharacterBrain.cs//控制的玩家
public Character ControlledCharacter;//攻击缓冲
public float AttackTimeout = 1f;
InputBuffer AttackBuffer;//绑定操作
void Start()
{GameInput.Instance.onAttack += () =>{AttackBuffer.Buffer(() => ControlledCharacter.Attack(), ECharacterState.Attack, AttackTimeout);};
}//轮询
void Update()
{AttackBuffer.Update();
}

Character的Attack函数实际上就是在调用角色状态机的Trigger函数,如下

//Character.cspublic void Attack()
{if (Equipment.CurrentWeapon == null)return;StateMachine.Trigger("Attack");
}

打断机制

我把打断机制写在了状态机中,作为一种扩展行为,在UnityHFSM中,一个State有一个needsExitTime字段,该字段为true时,除非强制转换,否则无法退出本状态,配合Animancer的动画事件(OnEnd)可以很好的让动画来控制状态的转换(我这样做是希望角色的各种动作更加真实,让逻辑和表现更容易统一)。

为此我们只需要新增一个叫做CanExitBeforeEnd的bool变量即可表示本状态是否可以被打断。于此同时,实际上完全没有必要为每个状态新建一个CanExitBeforeEnd变量,将该变量放在状态机类中,供每个状态变更,供外界访问即可。

现在我们只使用CanExitBeforeEnd表示状态希望被改变,但没有直接改变的逻辑,假设我们当前使用Trigger的方法进行状态变更,则我们只需要在每次Trigger前查询一次当前状态是否需要变更,需要变更且状态的needsExitTime为true,则将其设置为false,再执行Trigger操作即可。

StateMachine的Trigger操作并不是虚函数,只需要在源码中为其添加virtual关键字即可,然后在扩展的CharacterStateMachineBase中重写Trigger逻辑,代码如下

public class CharacterStateMachineData
{public bool CanExitBeforeEnd;
}public class CharacterStateMachineBase : StateMachine<ECharacterState>
{public Character Owner;public CharacterStateMachineData Data;public CharacterStateMachineBase(Character owner, CharacterStateMachineData data = null) : base(){Owner = owner;if(data != null )Data = data;elseData = new CharacterStateMachineData();}//分层状态机递归获取具体状态public CharacterStateBase CurrentState{get{if (ActiveState is CharacterStateMachineBase)return (ActiveState as CharacterStateMachineBase).CurrentState;return ActiveState as CharacterStateBase;}}public override void Trigger(string trigger){//允许玩家通过一些动作和输入打断当前状态if(ActiveState.needsExitTime && Data.CanExitBeforeEnd)ActiveState.needsExitTime = false;base.Trigger(trigger);}
}public class CharacterStateBase : State<ECharacterState>
{public Character Owner;new public CharacterStateMachineBase fsm => base.fsm as CharacterStateMachineBase;public virtual Vector3 DeltaMotion => Owner.Animancer.Animator.deltaPosition;public CharacterStateBase(Character owner) : base(){Owner = owner;}public override void OnEnter(){base.OnEnter();fsm.Data.CanExitBeforeEnd = false;}
}

以上是在使用Trigger时打断的操作,假设我们使用的是普通的Transition进行状态变更,那么是无法打断的,比如如下的移动操作,完全没有使用Trigger

MovementStateMachineBase GroundFSM = new MovementStateMachineBase(owner);
GroundFSM.AddState(EMovementState.Idle, new IdleState(owner));
GroundFSM.AddState(EMovementState.Move, new MoveState(owner));
GroundFSM.AddTwoWayTransition(EMovementState.Idle, EMovementState.Move,t => Owner.Parameters.MovementDirection.magnitude > 0);

为此,我们可以在期望被打断的状态机中特别指出何时可以被打断,我当前只在攻击状态时可以被打断,不打断会放完攻击动画,打断会立刻进入其他状态并播放动画

特别的代码如下,该代码会在AttackState的Update函数中被轮询

Parameters是Character的字段,用于与直接的GameInput解耦

private void UpdateInterrupt()
{if(!fsm.Data.CanExitBeforeEnd)return;//移动打断if (Owner.Parameters.MovementDirection.magnitude > 0){needsExitTime = false;}
}

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

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

相关文章

运维日记

运维 内容分发网络(CDN):使用CDN将图片分发到离用户更近的服务器,减少传输时间sudo的权限配置是通过/etc/sudoers文件管理的,这个文件定义了哪些用户在哪种条件下执行什么命令。直接编辑这个文件是有风险的,因为错误的语法可能导致系统无法正常使用sudo功能,因此使用vi…

前端需要学什么

作为前端开发者&#xff0c;学习的内容不仅仅局限于 HTML、CSS 和 JavaScript。前端技术栈发展迅速&#xff0c;想要在职业生涯中不断进步&#xff0c;可以从以下几个方面规划学习路径&#xff1a; 1. 核心前端技能 掌握前端开发的基础和进阶知识&#xff1a; HTML/CSS/JavaS…

用python+YOLOV8图片车辆车距

1. 检测车辆&#xff1a; 使用深度学习模型&#xff08;例如 YOLO、Mask R-CNN&#xff09;来检测照片中的车辆&#xff0c;并得到每辆车的边界框&#xff08;Bounding Box&#xff09;。 工具与技术&#xff1a; YOLOv5/YOLOv8&#xff1a;高效的实时目标检测模型。OpenCV&…

【数理统计】参数估计

文章目录 点估计矩估计法最大似然估计法 区间估计单个正态总体参数的区间估计均值 μ \mu μ 的区间估计方差 σ 2 \sigma^2 σ2 的区间估计 两个正态总体参数的区间估计&#xff08;略&#xff09;补充&#xff1a;单侧置信区间 点估计 矩估计法 【定义】设 X X X 是随机…

冯诺依曼架构与哈佛架构的对比与应用

冯诺依曼架构&#xff08;Von Neumann Architecture&#xff09;&#xff0c;也称为 冯诺依曼模型&#xff0c;是由著名数学家和计算机科学家约翰冯诺依曼&#xff08;John von Neumann&#xff09;在1945年提出的。冯诺依曼架构为现代计算机奠定了基础&#xff0c;几乎所有现代…

linux不同发行版中的主要差异

一、初始化系统 Linux不同发行版中的系统初始化系统&#xff08;如 System V init、Upstart 或 systemd&#xff09; System V init&#xff1a; 历史&#xff1a;System V init 是最传统的 Linux 系统初始化系统&#xff0c;起源于 Unix System V 操作系统。运行级别&#xff…

3D造型软件solvespace在windows下的编译

3D造型软件solvespace在windows下的编译 在逛开源社区的时候发现了几款开源CAD建模软件&#xff0c;一直囿于没有合适的建模软件&#xff0c;虽然了解了很多的模拟分析软件&#xff0c;却不能使之成为整体的解决方案&#xff0c;从而无法产生价值。opencascad之流虽然可行&…

机器学习04-为什么Relu函数

机器学习0-为什么Relu函数 文章目录 机器学习0-为什么Relu函数 [toc]1-手搓神经网络步骤总结2-为什么要用Relu函数3-进行L1正则化修改后的代码解释 4-进行L2正则化解释注意事项 5-Relu激活函数多有夸张1-细数Relu函数的5宗罪2-Relu函数5宗罪详述 6-那为什么要用这个Relu函数7-文…

Redis篇-21--运维篇3-集群(分片,水平扩展,高可用,集群配置案例,扩展哨兵案例)

1、概述 Redis集群&#xff08;Cluster&#xff09;通过分片&#xff08;sharding&#xff09;实现了水平扩展&#xff0c;允许数据分布在多个节点上&#xff0c;从而提升性能和存储容量。 在Redis集群中&#xff0c;数据被分割成16384个哈希槽&#xff08;hash slots&#x…

QScreen在Qt5.15与Qt6.8版本下的区别

简述 QScreen主要用于提供与屏幕相关的信息。它可以获取有关显示设备的分辨率、尺寸、DPI&#xff08;每英寸点数&#xff09;等信息。本文主要是介绍Qt5.15与Qt6环境下&#xff0c;QScreen的差异&#xff0c;以及如何判断高DPI设备。 属性说明 logicalDotsPerInch&#xff1…

[HNCTF 2022 Week1]你想学密码吗?

下载附件用记事本打开 把这些代码放在pytho中 # encode utf-8 # python3 # pycryptodemo 3.12.0import Crypto.PublicKey as pk from hashlib import md5 from functools import reducea sum([len(str(i)) for i in pk.__dict__]) funcs list(pk.__dict__.keys()) b reduc…

shell8

until循环(条件为假的时候一直循环和while相反) i0 until [ ! $i -lt 10 ] doecho $i((i)) done分析 初始化变量&#xff1a; i0&#xff1a;将变量i初始化为0。 条件判断 (until 循环)&#xff1a; until [ ! $i -lt 10 ]&#xff1a;这里的逻辑有些复杂。它使用了until循环…

【游戏中orika完成一个Entity的复制及其Entity异步落地的实现】 1.ctrl+shift+a是飞书下的截图 2.落地实现

一、orika工具使用 1)工具类 package com.xinyue.game.utils;import ma.glasnost.orika.MapperFactory; import ma.glasnost.orika.impl.DefaultMapperFactory;/*** author 王广帅* since 2022/2/8 22:37*/ public class XyBeanCopyUtil {private static MapperFactory mappe…

frp内网穿透笔记

文章目录 一、环境介绍二、配置过程2.1 下载文件2.3 服务器A的配置2.3 目标机B的配置2.4 电脑C怎么用 三、问题汇总 一、环境介绍 带公网的vps服务器A&#xff0c;需要穿透到的无公网目标电脑B&#xff0c;以及一台需要通过公网访问B的电脑C。frp 0.47.0&#xff1a;frp_0.47.…

PostgreSQL的交互式终端使用一系列命令来获取有关文本搜索配置对象的信息

在 psql&#xff08;PostgreSQL 的交互式终端&#xff09;中&#xff0c;你可以使用一系列命令来获取有关文本搜索配置对象的信息。这些命令主要围绕 \dF 系列&#xff0c;以及使用 SQL 查询 pg_ts_config 系统视图。以下是你可以使用的一些方法&#xff1a; 使用 \dF 系列命令…

写定制程序容易遇见的问题(FLASH不够时)

做了一个关于定制两条串口协议的活&#xff0c;主要是要在源代码基础上进行修改。源代码只剩了200多字节flash。本来最初我的想法很奇特&#xff0c;用结构体来模仿寄存器的写法。当我写完几行代码后&#xff0c;编译链接&#xff0c;立马就报CODE内存超了。 然后最终还是选择…

【Leetcode 热题 100 - 扩展】303. 区域和检索 - 数组不可变

问题背景 给定一个整数数组 n u m s nums nums&#xff0c;处理以下类型的多个查询&#xff1a; 计算索引 l e f t left left 和 r i g h t right right&#xff08;包含 l e f t left left 和 r i g h t right right&#xff09;之间的 n u m s nums nums 元素的 和 &a…

本地缓存和Redis缓存 存储更新时间的更新套路

//先获取redis key和local key //从reids中获取数据 – 为空 先设置redis缓存30天,value为当前时间 然后设置本地缓存,value为当前时间 从数据库里读数据 – 不为空 获取本地缓存时间 if本地缓存时间 < redis缓存时间(认为已更新)或者本地为空 从数据库读数据 else 从本地缓…

处理错误的两种方式:try...catch 与 then...catch

一、try...catch try...catch 是一种用于捕获和处理同步代码中异常的机制。其基本结构如下&#xff1a; try {// 可能会抛出异常的代码 } catch (error) {// 处理异常 }使用场景&#xff1a; 主要用于同步代码&#xff0c;尤其是在需要处理可能抛出的异常时。适用于函数调用…

【十进制整数转换为其他进制数——短除形式的贪心算法】

之前写过一篇用贪心算法计算十进制转换二进制的方法&#xff0c;详见&#xff1a;用贪心算法计算十进制数转二进制数&#xff08;整数部分&#xff09;_短除法求二进制-CSDN博客 经过一段时间的研究&#xff0c;本人又发现两个规律&#xff1a; 1、不仅仅十进制整数转二进制可…