当前位置: 首页 > news >正文

【《游戏编程模式》实战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
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;// 阈值Apublic int thresholdB=20;// 阈值Bpublic 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
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的类不能直接newIdleState = 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 ,该胶囊体切换至死亡状态后执行销毁行为,另一胶囊体不受影响。

相关文章:

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

目录 1、状态模式 2、使用工具 3、状态模式适用范围 4、实现内容 5、代码及思路 Enemy.cs EnemyState.cs 6、unity里的设置 7、运行效果展示 1、状态模式 “允许一个对象在其内部状态改变时改变自身的行为。对象看起来好像是在修改自身类。” 就是一个对象能随着自己…...

借助免费GIS工具箱轻松实现las点云格式到3dtiles格式的转换

在当今数字化浪潮下&#xff0c;地理信息系统&#xff08;GIS&#xff09;技术日新月异&#xff0c;广泛渗透到城市规划、地质勘探、文化遗产保护等诸多领域。而 GISBox 作为一款功能强大且易用的 GIS 工具箱&#xff0c;以轻量级、免费使用、操作便捷等诸多优势&#xff0c;为…...

科研绘图系列:R语言科研绘图之标记热图(heatmap)

禁止商业或二改转载,仅供自学使用,侵权必究,如需截取部分内容请后台联系作者! 文章目录 介绍加载R包数据下载导入数据数据预处理画图系统信息参考介绍 科研绘图系列:R语言科研绘图之标记热图(heatmap) 加载R包 library(tidyverse) library(ggplot2) library(reshape)…...

【轻松学C:编程小白的大冒险】--- C语言简介 02

在编程的艺术世界里&#xff0c;代码和灵感需要寻找到最佳的交融点&#xff0c;才能打造出令人为之惊叹的作品。而在这座秋知叶i博客的殿堂里&#xff0c;我们将共同追寻这种完美结合&#xff0c;为未来的世界留下属于我们的独特印记。 【轻松学C&#xff1a;编程小白的大冒险】…...

《HeadFirst设计模式》笔记(上)

设计模式的目录&#xff1a; 1 设计模式介绍 要不断去学习如何利用其它开发人员的智慧与经验。学习前人的正统思想。 我们认为《Head First》的读者是一位学习者。 一些Head First的学习原则&#xff1a; 使其可视化将文字放在相关图形内部或附近&#xff0c;而不是放在底部…...

数据结构:ArrayList与顺序表

目录 &#x1f4d6;一、什么是List &#x1f4d6;二、线性表 &#x1f4d6;三、顺序表 &#x1f42c;1、display()方法 &#x1f42c;2、add(int data)方法 &#x1f42c;3、add(int pos, int data)方法 &#x1f42c;4、contains(int toFind)方法 &#x1f42c;5、inde…...

SpringBoot之核心配置

学习目标&#xff1a; 1.熟悉Spring Boot全局配置文件的使用 2.掌握Spring Boot配置文件属性值注入 3.熟悉Spring Boot自定义配置 4.掌握Profile多环境配置 5.了解随机值设置以及参数间引用 1.全局配置文件 Spring Boot使用 application.properties 或者application.yaml 的文…...

EasyExcel上传校验文件错误信息放到文件里以Base64 返回给前端

产品需求&#xff1a; 前端上传个csv 或 excel 文件&#xff0c;文件共4列&#xff0c;验证文件大小&#xff0c;类型&#xff0c;文件名长度&#xff0c;文件内容&#xff0c;如果某行某个单元格数据验证不通过&#xff0c;就把错误信息放到这行第五列&#xff0c;然后把带有…...

单片机软件定时器V4.0

单片机软件定时器V4.0 用于单片机定时执行任务等&#xff0c;比如LED GPIO等定时控制&#xff0c;内置前后台工作模式 头文件有使用例子 #ifndef __SORFTIME_APP_H #define __SORFTIME_APP_H#ifdef __cplusplus extern "C" { #endif#include <stdint.h>// #…...

超完整Docker学习记录,Docker常用命令详解

前言 关于国内拉取不到docker镜像的问题&#xff0c;可以利用Github Action将需要的镜像转存到阿里云私有仓库&#xff0c;然后再通过阿里云私有仓库去拉取就可以了。 参考项目地址&#xff1a;使用Github Action将国外的Docker镜像转存到阿里云私有仓库 一、Docker简介 Do…...

C++ 入门第26天:文件与流操作基础

往期回顾&#xff1a; C 入门第23天&#xff1a;Lambda 表达式与标准库算法入门-CSDN博客 C 入门第24天&#xff1a;C11 多线程基础-CSDN博客 C 入门第25天&#xff1a;线程池&#xff08;Thread Pool&#xff09;基础-CSDN博客 C 入门第26天&#xff1a;文件与流操作基础 前言…...

使用python将多个Excel表合并成一个表

import pandas as pd# 定义要合并的Excel文件路径和名称 file_paths [file1.xlsx, file2.xlsx, file3.xlsx, file4.xlsx, file5.xlsx]# 创建一个空的DataFrame来存储合并后的数据 merged_data pd.DataFrame()# 循环遍历每个Excel文件&#xff0c;并读取其中的数据 for file_p…...

halcon三维点云数据处理(七)find_shape_model_3d_recompute_score

目录 一、find_shape_model_3d_recompute_score例程代码二、set_object_model_3d_attrib_mod函数三、prepare_object_model_3d 函数四、create_cube_shape_model_3d函数五、获得CamPose六、project_cube_image函数七、find_shape_model_3d函数八、project_shape_model_3d函数 一…...

vue js实现时钟以及刻度效果

2025.01.08今天我学习如何用js实现时钟样式&#xff0c;效果如下&#xff1a; 一、html代码如下&#xff1a; <template><!--圆圈--><div class"notice_border"><div class"notice_position notice_name_class" v-for"item in …...

unity学习15:预制体prefab

目录 1 创建多个gameobject 2 创建prefab 2.1 创建prefab &#xff08;类&#xff09; 2.2 prefab 是一个文件 2.3 prefab可以导出 3 创建prefab variant &#xff08;子类&#xff09; 3.1 除了创建多个独立的prefab&#xff0c; 还可以创建 prefab variant 3.2 他…...

基于Thinkphp6+uniapp的陪玩陪聊软件开发方案分析

使用uni-app框架进行前端开发。uni-app是一个使用Vue.js开发所有前端应用的框架&#xff0c;支持一次编写&#xff0c;多端发布&#xff0c;包括APP、小程序、H5等。 使用Thinkphp6框架进行后端开发。Thinkphp6是一个轻量级、高性能、面向对象的PHP开发框架&#xff0c;具有易…...

MySQL - 子查询和相关子查询详解

在SQL中&#xff0c;子查询&#xff08;Subquery&#xff09;和相关子查询&#xff08;Correlated Subquery&#xff09;是非常强大且灵活的工具&#xff0c;可以用于执行复杂的数据检索和操作。它们允许我们在一个查询中嵌套另一个查询&#xff0c;从而实现更复杂的逻辑和条件…...

Android 系统签名 keytool-importkeypair

要在 Android 项目中使用系统签名并将 APK 打包时与项目一起打包&#xff0c;可以按照以下步骤操作&#xff1a; 步骤 1&#xff1a;准备系统签名文件 从 Android 系统源码中获取系统签名文件&#xff0c;通常位于 build/target/product/security 目录下&#xff0c;包括 pla…...

安卓漏洞学习(十八):Android加固基本原理

APP加固技术发展历程 APK加固整体思路 加固整体思路&#xff1a;先解压apk文件&#xff0c;取出dex文件&#xff0c;对dex文件进行加密&#xff0c;然后组合壳中的dex文件&#xff08;Android类加载机制&#xff09;&#xff0c;结合之前的apk资源&#xff08;解压apk除dex以外…...

Docker 使用Dockerfile创建镜像

创建并且生成镜像 在当前目录下创建一个名为Dockerfile文件 vi Dockerfile填入下面配置 # 使用 CentOS 作为基础镜像 FROM centos:7# 设置工作目录 WORKDIR /app# 复制项目文件到容器中 COPY bin/ /app/bin/ COPY config/ /app/config/ COPY lib/ /app/lib/ COPY plugin/ /a…...

【OSG学习笔记】Day 18: 碰撞检测与物理交互

物理引擎&#xff08;Physics Engine&#xff09; 物理引擎 是一种通过计算机模拟物理规律&#xff08;如力学、碰撞、重力、流体动力学等&#xff09;的软件工具或库。 它的核心目标是在虚拟环境中逼真地模拟物体的运动和交互&#xff0c;广泛应用于 游戏开发、动画制作、虚…...

DIY|Mac 搭建 ESP-IDF 开发环境及编译小智 AI

前一阵子在百度 AI 开发者大会上&#xff0c;看到基于小智 AI DIY 玩具的演示&#xff0c;感觉有点意思&#xff0c;想着自己也来试试。 如果只是想烧录现成的固件&#xff0c;乐鑫官方除了提供了 Windows 版本的 Flash 下载工具 之外&#xff0c;还提供了基于网页版的 ESP LA…...

NFT模式:数字资产确权与链游经济系统构建

NFT模式&#xff1a;数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新&#xff1a;构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议&#xff1a;基于LayerZero协议实现以太坊、Solana等公链资产互通&#xff0c;通过零知…...

SpringCloudGateway 自定义局部过滤器

场景&#xff1a; 将所有请求转化为同一路径请求&#xff08;方便穿网配置&#xff09;在请求头内标识原来路径&#xff0c;然后在将请求分发给不同服务 AllToOneGatewayFilterFactory import lombok.Getter; import lombok.Setter; import lombok.extern.slf4j.Slf4j; impor…...

VM虚拟机网络配置(ubuntu24桥接模式):配置静态IP

编辑-虚拟网络编辑器-更改设置 选择桥接模式&#xff0c;然后找到相应的网卡&#xff08;可以查看自己本机的网络连接&#xff09; windows连接的网络点击查看属性 编辑虚拟机设置更改网络配置&#xff0c;选择刚才配置的桥接模式 静态ip设置&#xff1a; 我用的ubuntu24桌…...

基于SpringBoot在线拍卖系统的设计和实现

摘 要 随着社会的发展&#xff0c;社会的各行各业都在利用信息化时代的优势。计算机的优势和普及使得各种信息系统的开发成为必需。 在线拍卖系统&#xff0c;主要的模块包括管理员&#xff1b;首页、个人中心、用户管理、商品类型管理、拍卖商品管理、历史竞拍管理、竞拍订单…...

Web后端基础(基础知识)

BS架构&#xff1a;Browser/Server&#xff0c;浏览器/服务器架构模式。客户端只需要浏览器&#xff0c;应用程序的逻辑和数据都存储在服务端。 优点&#xff1a;维护方便缺点&#xff1a;体验一般 CS架构&#xff1a;Client/Server&#xff0c;客户端/服务器架构模式。需要单独…...

高分辨率图像合成归一化流扩展

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 1 摘要 我们提出了STARFlow&#xff0c;一种基于归一化流的可扩展生成模型&#xff0c;它在高分辨率图像合成方面取得了强大的性能。STARFlow的主要构建块是Transformer自回归流&#xff08;TARFlow&am…...

李沐--动手学深度学习--GRU

1.GRU从零开始实现 #9.1.2GRU从零开始实现 import torch from torch import nn from d2l import torch as d2l#首先读取 8.5节中使用的时间机器数据集 batch_size,num_steps 32,35 train_iter,vocab d2l.load_data_time_machine(batch_size,num_steps) #初始化模型参数 def …...

【版本控制】GitHub Desktop 入门教程与开源协作全流程解析

目录 0 引言1 GitHub Desktop 入门教程1.1 安装与基础配置1.2 核心功能使用指南仓库管理日常开发流程分支管理 2 GitHub 开源协作流程详解2.1 Fork & Pull Request 模型2.2 完整协作流程步骤步骤 1: Fork&#xff08;创建个人副本&#xff09;步骤 2: Clone&#xff08;克隆…...