Unity学习之FSM有限状态机
作者:念丶小宇 时间:2022-09-10 06:18:33
前言:一个游戏里的一个人物会存在多种状态,那么就需要有一个专门管理这些状态的类。不然会显得杂乱无章,不易于后面状态的增加或者减少。
思路:既然要方便管理,那么首先肯定得有个系统类(专门用来存放所有的状态、状态的增删等功能);然后就是需要把所有的状态都单独写一个类(已达到修改某个状态的时候,其他状态不会受到影响)。
状态管理类:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class FSMSystem
{
private Dictionary<StateID, FSMState> states = new Dictionary<StateID, FSMState>();
private StateID currentStateID;
private FSMState currentFSMState;
public void Update(GameObject npc)
{
currentFSMState.Act(npc);
currentFSMState.Reason(npc);
}
/// <summary>
/// 添加状态
/// </summary>
/// <param name="fSMState"></param>
public void AddState(FSMState fSMState)
{
if (fSMState == null) return;
//if (currentFSMState == null)
//{
currentStateID = fSMState.ID;
currentFSMState = fSMState;
//}
if (states.ContainsKey(currentStateID)) return;
states.Add(currentStateID, currentFSMState);
}
/// <summary>
/// 删除状态
/// </summary>
/// <param name="stateID"></param>
public void DeleteState(StateID stateID)
{
if (stateID == StateID.Null) return;
if (!states.ContainsKey(stateID)) return;
states.Remove(stateID);
}
/// <summary>
/// 执行状态条件转换
/// </summary>
/// <param name="transition"></param>
public void PerformTransition(Transition transition)
{
if (transition == Transition.NullTransition) return;
StateID stateID = currentFSMState.GetStateID(transition);
if (stateID == StateID.Null) return;
if (!states.ContainsKey(stateID)) return;
FSMState fSMState = states[stateID];
currentFSMState.StateExit();
currentFSMState = fSMState;
currentStateID = fSMState.ID;
currentFSMState.StateEnter();
}
}
状态基类:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public enum Transition
{
NullTransition,
SeePlayer,//发现玩家
LostPlayer,//玩家脱离视野范围
AttackPlayer,//攻击玩家
}
public enum StateID
{
Null,
Chase,//追逐
Patrol,//巡逻
Attack,//攻击
}
public abstract class FSMState
{
protected Transition transition;
protected StateID stateID;
protected FSMSystem fSM;
public StateID ID
{
get { return stateID; }
}
protected Dictionary<Transition, StateID> dic = new Dictionary<Transition, StateID>();
public FSMState(FSMSystem fSM)
{
this.fSM = fSM;
}
/// <summary>
/// 增加状态
/// </summary>
/// <param name="transition"></param>
/// <param name="stateID"></param>
public void AddTransition(Transition transition, StateID stateID)
{
if (transition == Transition.NullTransition) return;
if (stateID == StateID.Null) return;
if (dic.ContainsKey(transition)) return;
dic.Add(transition, stateID);
}
/// <summary>
/// 删除状态
/// </summary>
/// <param name="transition"></param>
public void DeleteTransition(Transition transition)
{
if (transition == Transition.NullTransition) return;
if (!dic.ContainsKey(transition)) return;
dic.Remove(transition);
}
/// <summary>
/// 获取状态
/// </summary>
/// <param name="transition"></param>
/// <returns></returns>
public StateID GetStateID(Transition transition)
{
if (transition == Transition.NullTransition) return StateID.Null;
if (!dic.ContainsKey(transition)) return StateID.Null;
return dic[transition];
}
/// <summary>
/// 进入状态
/// </summary>
public virtual void StateEnter() { }
/// <summary>
/// 退出状态
/// </summary>
public virtual void StateExit() { }
/// <summary>
/// 状态持续中,,,
/// </summary>
/// <param name="npc"></param>
public abstract void Act(GameObject npc);
/// <summary>
/// 状态退出前,,,
/// </summary>
/// <param name="npc"></param>
public abstract void Reason(GameObject npc);
}
巡逻状态:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
/// <summary>
/// 巡逻状态
/// </summary>
public class PatrolState : FSMState
{
/// <summary>
/// 巡逻路径点集合
/// </summary>
private Transform[] paths;
/// <summary>
/// 当前巡逻路径点索引
/// </summary>
private int index = 0;
/// <summary>
/// 移动速度
/// </summary>
private float moveSpeed = 0.5f;
/// <summary>
/// 玩家
/// </summary>
private Transform player;
public PatrolState(FSMSystem fSM, Transform player) : base(fSM)
{
this.player = player;
paths = GameObject.Find("Path").GetComponentsInChildren<Transform>();
stateID = StateID.Patrol;
}
public override void Act(GameObject npc)
{
npc.transform.LookAt(paths[index].position);
npc.transform.Translate(Vector3.forward * Time.deltaTime * moveSpeed);
if (Vector3.Distance(npc.transform.position, paths[index].position) < 1)
{
index++;
index %= paths.Length;
}
}
public override void Reason(GameObject npc)
{
npc.GetComponent<Animator>().SetFloat("Speed", moveSpeed );
if (Vector3.Distance(player.position, npc.transform.position) < 10)
{
fSM.PerformTransition(Transition.SeePlayer);
}
}
}
追逐状态:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
/// <summary>
/// 追逐状态
/// </summary>
public class ChaseState : FSMState
{
/// <summary>
/// 移动速度
/// </summary>
private float moveSpeed = 2f;
/// <summary>
/// 玩家
/// </summary>
private Transform player;
public ChaseState(FSMSystem fSM, Transform player) : base(fSM)
{
stateID = StateID.Chase;
this. player = player;
}
public override void Act(GameObject npc)
{
npc.transform.LookAt(player.position);
npc.transform.Translate(Vector3.forward * Time.deltaTime * moveSpeed);
}
public override void Reason(GameObject npc)
{
npc.GetComponent<Animator>().SetFloat("Speed", moveSpeed / 2);
if (Vector3.Distance(player.position, npc.transform.position) >= 10)
{
fSM.PerformTransition(Transition.LostPlayer);
}
else if (Vector3.Distance(player.position, npc.transform.position) <= 1f )
{
fSM.PerformTransition(Transition.AttackPlayer);
}
}
}
攻击状态:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
/// <summary>
/// 攻击状态
/// </summary>
public class AttackState : FSMState
{
/// <summary>
/// 玩家
/// </summary>
private Transform player;
public AttackState(FSMSystem fSM, Transform player) : base(fSM)
{
stateID = StateID.Attack;
this.player = player;
}
public override void Act(GameObject npc)
{
}
public override void Reason(GameObject npc)
{
if (Vector3.Distance(player.position, npc.transform.position) > 1f)
{
if (Vector3.Distance(player.position, npc.transform.position) >= 10)
{
fSM.PerformTransition(Transition.LostPlayer);
}
else if (Vector3.Distance(player.position, npc.transform.position) < 10)
{
fSM.PerformTransition(Transition.SeePlayer);
}
return;
}
npc.GetComponent<Animator>().SetTrigger("Attack01");
}
}
状态持有者实现类:
using System.Collections;using System.Collections.Generic;using UnityEngine;public class Enemy : MonoBehaviour{ private FSMSystem fSM; private Transform player; private void Start() { fSM = new FSMSystem(); FSMState patrolState = new PatrolState(fSM, player); patrolState.AddTransition(Transition.SeePlayer, StateID.Chase); patrolState.AddTransition(Transition.AttackPlayer, StateID.Attack); //patrolState.AddTransition(Transition.LostPlayer, StateID.Patrol); FSMState chaseState = new ChaseState(fSM, player); chaseState.AddTransition(Transition.LostPlayer, StateID.Patrol); chaseState.AddTransition(Transition.AttackPlayer, StateID.Attack); //chaseState.AddTransition(Transition.SeePlayer, StateID.Chase); FSMState attackState = new AttackState(fSM, player); attackState.AddTransition(Transition.SeePlayer, StateID.Chase); attackState.AddTransition(Transition.LostPlayer, StateID.Patrol); //attackState.AddTransition(Transition.AttackPlayer, StateID.Attack); fSM.AddState(patrolState); fSM.AddState(chaseState); fSM.AddState(attackState); } private void Update() { fSM.Update(gameObject); }}
来源:https://blog.csdn.net/qq_43815828/article/details/117918980
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
Java Spring之@Async原理案例详解
WebService教程详解(二)
详解Java中final的用法
Java接口的作用_动力节点Java学院整理
![](https://img.aspxhome.com/file/2023/0/61570_0s.png)
C#在Winform开发中使用Grid++报表
![](https://img.aspxhome.com/file/2023/1/113711_0s.png)
Android使用SharedPreferences存储数据的实现方法
深入浅出讲解Java集合之Collection接口
![](https://img.aspxhome.com/file/2023/2/89622_0s.png)
高效C#编码优化原则
Springboot JPA 枚举Enum类型存入到数据库的操作
![](https://img.aspxhome.com/file/2023/4/60064_0s.jpg)
详解Java中String类的各种用法
![](https://img.aspxhome.com/file/2023/0/91730_0s.png)
详解Spring与Mybatis整合方法(基于IDEA中的Maven整合)
![](https://img.aspxhome.com/file/2023/0/58690_0s.png)
Android开发之使用ViewPager实现图片左右滑动切换效果
Java实现堆排序(大根堆)的示例代码
Springboot引入hibernate配置自动建表并进行增删改查操作
![](https://img.aspxhome.com/file/2023/9/76499_0s.png)
Java中方法重写与重载的区别
Android程序开发中单选按钮(RadioGroup)的使用详解
![](https://img.aspxhome.com/file/2023/5/137955_0s.png)
C#使用FolderBrowserDialog类实现选择打开文件夹方法详解
Android中复制图片的实例代码
C# 实现绘制PDF嵌套表格案例详解
![](https://img.aspxhome.com/file/2023/1/83311_0s.png)