【《游戏编程模式》实战04】状态模式实现敌人AI

目录

1、状态模式

2、使用工具

3、状态模式适用范围

4、实现内容

5、代码及思路

Enemy.cs

EnemyState.cs

6、unity里的设置

7、运行效果展示


1、状态模式

"允许一个对象在其内部状态改变时改变自身的行为。对象看起来好像是在修改自身类。"

就是一个对象能随着自己的状态改变执行不同的方法吧,而这个方法和状态并不是写在这个对象类里的,却能达到像是对象修改自身类的效果。

2、使用工具

Unity2022.3.51f1c1、visual studio code

3、状态模式适用范围

1)你有一个游戏实体,它的行为 基于它的内部状态改变

2)这些状态被严格分为相对数目较少的小集合

3)游戏实体随着时间的变化会响应用户输入 或一些游戏事件

在游戏里广泛使用在ai里,也经常被应用于用户输入处理、浏览菜单屏幕、解析文件、网络协议和其他异步的行为

4、实现内容

实现敌人能在这几个状态之间根据距离远近自动切换,hp为0时死亡

三段距离:一段距离内攻击,一段距离内追逐,更远的距离会暂停

但是暂停、防御的触发、检验技能是否释放完毕等功能并没有实现,这里只是搭框架,原理都差不多,只要学会一个条件的状态转移其他的都很好做,就不赘述了。

5、代码及思路

书上的代码看着太复杂了,写的时候没有完全参考,我更倾向于围绕状态模式的目标------将每个状态相关 的所有数据和行为封装对应状态类里面 来写。

状态模式的基本模式:参考状态模式 | 菜鸟教程

  • 上下文类(Enemy):它持有一个状态引用 ,并在状态改变时更新行为
  • 状态类(EnemyState):定义所有 可能的状态 并为它们提供行为接口
  • 具体状态类(IdleState\StandState...):实现状态接口的具体类,表示对象的某一具体状态

具体来说在我的代码里结构是这样的

Enemy.cs
cs 复制代码
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Events;
// 每个敌人身上都该有
public class Enemy : MonoBehaviour
{
    [HideInInspector]public EnemyState state;
    public int distance;// 距离
    public int thresholdA=10;// 阈值A
    public int thresholdB=20;// 阈值B
    public int hp=10;// 血量
    UnityEvent<Enemy> stateChangeEvent = new UnityEvent<Enemy>();//状态改变事件
    void Start()
    {
        ChangeState(EnemyState.IdleState);//设置为初始状态
    }
    
    public void ChangeState(EnemyState newState)
    {
        if (state != null)
        {
            stateChangeEvent.RemoveListener(state.handleData);//移除原本的方法引用
        }
        state = newState;//切换至新状态
        stateChangeEvent.AddListener(state.handleData);//添加新状态的方法引用
        stateChangeEvent.Invoke(this);//调用下一个状态的handleData
    }
    
}
  • 在Start方法中初始化了敌人的状态并调用初始状态的handleData方法,这样在开始时,敌人就会执行与IdleState状态相关的行为。
  • 当敌人的状态改变时,ChangeState方法通过 stateChangeEvent.Invoke(this)触发事件,从而调用切换后状态的 handleData方法。(有一个坑是在使用AddListener的时候里面的参数传的其实是一个方法引用(地址值),因此在改变state的值以前,一定要把原本注册的方法移除,再添加新的,不然调用的还会是之前状态的方法/空引用异常。)

这本书这部分写得真不咋滴啊,对象类的函数不清不楚,没写如果把条件判断也加到状态类里去了,对象要怎么切换状态,这部分是我自己想的。

EnemyState.cs
cs 复制代码
using System.Collections;
using System.Collections.Generic;
using Unity.VisualScripting;
using Unity.VisualScripting.Antlr3.Runtime;
using UnityEditor;
using UnityEditor.EditorTools;
using UnityEngine;

public class EnemyState :MonoBehaviour
{
    // 定义所有可能的状态
    public static IdleState IdleState; // 空闲状态
    public static StandState StandState; // 站立状态
    public static MoveState MoveState; // 移动状态
    public static AttackState AttackState; // 攻击状态
    public static DieState DieState; // 死亡状态
    public static DefenseState DefenseState;// 防御状态
    void Awake()
    {
        // Awake方法在对象被激活时调用且会被派生类继承
        // 派生类生成时也会被调用,所以一定要加条件防止重复添加
        if (gameObject.GetComponent<IdleState>() == null)
        {
            // 添加各个状态的组件实例
            // 用add的原因是继承自MonoBehaviour的类不能直接new
            IdleState = gameObject.AddComponent<IdleState>();
            StandState = gameObject.AddComponent<StandState>();
            MoveState = gameObject.AddComponent<MoveState>();
            AttackState = gameObject.AddComponent<AttackState>();
            DieState = gameObject.AddComponent<DieState>();
            DefenseState = gameObject.AddComponent<DefenseState>();
        }
    }
    
    // 虚拟的handleData方法,用于每个状态具体的处理逻辑
    public virtual void HandleData(Enemy enemy){
        
        // 检查敌人的血量
        StartCoroutine(UpdateDieState(enemy));
      
    }

    // 虚拟的状态更新方法,每个状态可以自定义自己的更新逻辑
    public virtual IEnumerator UpdateState(Enemy enemy)
    {
        yield return null;
    }

    // 死亡状态的更新方法,持续检查敌人的血量是否小于等于零
    public  IEnumerator UpdateDieState(Enemy enemy)//这个只需要在初始状态调一次
    {
        while (true)
        {
            if (enemy.hp <= 0)
            {
                enemy.ChangeState(EnemyState.DieState);
                break;
            }
            yield return null;
        }
           
        }
    
}
public class IdleState :EnemyState
{
    public override void HandleData(Enemy enemy)
    {
        base.HandleData(enemy);// 调用基类的处理方法,检查死亡状态
        Debug.Log("初始状态");
        enemy.ChangeState(EnemyState.StandState);
    }
}
public class StandState : EnemyState
{
   
    public override void HandleData(Enemy enemy)
    {

        Debug.Log("站立状态");
        StartCoroutine(UpdateState(enemy));// 开始更新状态,判断是否切换到其他状态
    }
    public override IEnumerator UpdateState(Enemy enemy)
    {
        while (true)
        {
            if (enemy.distance < enemy.thresholdA) // 距离小于阈值A,进入攻击状态
            {
                enemy.ChangeState(EnemyState.AttackState);
                yield break; // 结束当前协程
            }
            else if (enemy.distance > enemy.thresholdA) // 距离大于阈值A,进入移动状态
            {
                enemy.ChangeState(EnemyState.MoveState);
                yield break; // 结束当前协程
            }
            yield return null; // 每帧检查一次
        }
    }
}
public class MoveState : EnemyState
{
     public override void HandleData(Enemy enemy)
    {
        Debug.Log("追逐状态");
        StartCoroutine(UpdateState(enemy)); // 开始更新状态,判断是否切换到其他状态
    }

    public override IEnumerator UpdateState(Enemy enemy)
    {
        while (true)
        {
            if (enemy.distance < enemy.thresholdA) // 距离小于阈值A,进入站立状态
            {
                enemy.ChangeState(EnemyState.StandState);
                break; // 结束当前协程
            }
            yield return null; // 每帧检查一次
        }
    }
    
}
public class AttackState : EnemyState
{
    public override void HandleData(Enemy enemy)
    {
       
        Debug.Log("攻击状态");
        StartCoroutine(UpdateState(enemy));// 开始更新状态,判断是否切换到其他状态

    }
    public override IEnumerator UpdateState(Enemy enemy)
    {
        while (true)
        {
            if (enemy.distance > enemy.thresholdA) // 距离大于阈值A,进入移动状态
            {
                enemy.ChangeState(EnemyState.MoveState);
                break; // 结束当前协程
            }
            yield return null; // 每帧检查一次
        }
    }
}
public class DieState : EnemyState
{
    public override void HandleData(Enemy enemy)
    {
        GameObject.Destroy(enemy.gameObject);// 销毁敌人对象,表示死亡
        Debug.Log("死亡状态");
    }
}
public class DefenseState : EnemyState
{
    public override void HandleData(Enemy enemy)
    {
      
        Debug.Log("防御状态");
        StartCoroutine(UpdateState(enemy));// 开始更新状态,判断是否切换到其他状态
 
    }
    public override IEnumerator UpdateState(Enemy enemy)
    {
        while (true)
        {
            //如果是释放过程中注意都要等防御技能结束才行
            //防御状态要切换到其他状态的条件应是被打断/成功防住
            if (enemy.distance > enemy.thresholdA)// 距离大于阈值A,进入移动状态
            {
                enemy.ChangeState(EnemyState.MoveState);
                break;// 结束当前协程
            }
            else if (enemy.distance < enemy.thresholdA)// 距离小于阈值A,进入攻击状态
            {
                enemy.ChangeState(EnemyState.AttackState);
                break;// 结束当前协程
            }
            yield return null;// 每帧检查一次
        }
    }
}
  • 通过EnemyState 作为基类来定义所有可能的敌人状态(如 IdleState、MoveState、AttackState 等)。每个状态类都会实现 handleData方法和 UpdateState 协程方法,这样每个状态都可以自定义自己的行为和状态转移条件。
  • 状态切换:通过 enemy.ChangeState(EnemyState.XXX) 来切换敌人的状态,每个状态都有自己的逻辑来判断何时切换到下一个状态。
  • 血量和死亡判断:EnemyState类中的 UpdateDieState 协程用于实时检查敌人的血量,一旦血量降至 0 或以下,就会触发死亡状态并销毁敌人对象。
  • 状态更新:handleData 方法负责初始化和启动状态检查,UpdateState 协程负责检查条件并更新状态。
6、unity里的设置

建一个空物体挂上EnemyState类(继承自monobehavior类的都需要挂到某物体上才能用)

建两个胶囊体挂上Enemy类,阈值和血量可以根据敌人的不同自定义,这里我随便设的值。

7、运行效果展示

由于这只是框架,所以我检查效果的方式就是直接在Inspector里调Distance来观察状态转换是否成功。对照着这个状态转移图来看:

运行后,两个胶囊体分别从初始进入站立状态,又由于d<a而进入攻击状态

将其中一个胶囊体的Distance调为11>a,攻击->追逐。另外可观察到另一个胶囊体状态不受影响

将胶囊体Distance调回0,按照状态转移图从追逐->站立->攻击

最后,将胶囊体的Hp调为0 ,该胶囊体切换至死亡状态后执行销毁行为,另一胶囊体不受影响。

相关推荐
影音小博士6 分钟前
分享几个高清无水印国外视频素材网站
经验分享·学习·音视频
Rinai_R20 分钟前
MySQL学习记录1【DQL和DCL】
数据库·笔记·学习·mysql
s_little_monster25 分钟前
【Linux】Linux常见指令(上)
linux·运维·服务器·经验分享·笔记·学习·centos
Damon小智29 分钟前
C#进阶-在Ubuntu上部署ASP.NET Core Web API应用
linux·nginx·c#·asp.net·.net·.net core
csdn_aspnet32 分钟前
C# 或 .NetCore 如何使用 NPOI 导出图片到 Excel 文件
c#·excel·.netcore
纪伊路上盛名在1 小时前
从视频中截取ppt,整理为pdf
笔记·学习·计算机视觉·pdf·powerpoint·音视频·学习方法
PassionY2 小时前
Unity-Mirror网络框架-从入门到精通之AdditiveScenes 示例
unity·unet·photon·networkmanager·mirror·多人联网·ngo
灵性(๑>ڡ<)☆2 小时前
Vue3学习-day2
前端·vue.js·学习
大丈夫立于天地间2 小时前
OSPF - 特殊区域
网络·网络协议·学习·算法·信息与通信
LuckyLay2 小时前
Golang学习笔记_22——Reader示例
笔记·学习·golang·reader·io.reader