Unity自学之旅05

Unity自学之旅05

  • Unity学习之旅⑤
  • 📝 AI基础与敌人行为
    • 🥊 AI导航
      • 理论知识(基础)
      • 开始实践
    • 🎃 敌人游戏机制
      • 追踪玩家
      • 攻击玩家
      • 子弹碰撞
      • 完善游戏失败条件
  • 🤗 总结归纳

Unity学习之旅⑤

📝 AI基础与敌人行为

🥊 AI导航

理论知识(基础)

一、导航系统的基本组件

  1. NavMesh(导航网格)
    • NavMesh 是 Unity 中 AI 导航的基础,它是一个表示可行走区域的网格。它将场景中的可行走表面(如地面、楼梯等)进行三角剖分,形成一个平面的网格,用于代理(Agent)导航。
    • 生成 NavMesh 的步骤:
      • 首先,需要将场景中的几何体标记为静态(Static),以便 Unity 可以将它们纳入 NavMesh 的计算。
      • 然后,打开 Navigation 窗口(Window -> AI -> Navigation),在 Bake 选项卡中调整设置,例如 Agent Radius(代理半径)、Agent Height(代理高度)等,这些设置决定了代理在场景中占用的空间大小和形状。
      • 点击 Bake 按钮,Unity 会生成 NavMesh,显示为场景中的蓝色网格。
  2. NavMeshAgent(导航网格代理)
    • 这是附加在游戏对象上的组件,用于让对象在 NavMesh 上移动。
    • 主要属性:
      • Speed:代理的移动速度。
      • Acceleration:代理的加速度。
      • Stopping Distance:距离目标多远时停止移动。
      • Radius:代理的半径,用于避障和计算可行走区域。
      • Height:代理的高度,决定了代理可以通过的最小空间高度。
  3. NavMeshObstacle(导航网格障碍物)
    • 该组件用于表示场景中的动态障碍物。当一个对象附加了 NavMeshObstacle 时,它会影响 NavMeshAgent 的导航路径。
    • 可以设置为 Carve 模式,这会在 NavMesh 上动态地修改 NavMesh,使代理能够绕开障碍物。

二、导航行为和逻辑

  1. 路径规划
    • NavMeshAgent 会自动根据 NavMesh 为代理规划从当前位置到目标位置的最短路径。它会考虑障碍物和可行走区域。
    • 可以使用 agent.remainingDistance 来检查代理离目标的剩余距离,使用 agent.pathStatus 来查看路径的状态,如是否完成、正在计算或被阻塞。
  2. 避障
    • Unity 的导航系统会自动处理代理之间和代理与障碍物之间的避障。通过设置 NavMeshAgent 的属性,如 RadiusAvoidance Priority,可以调整避障行为。
    • 代理之间会根据彼此的 Avoidance Priority 进行避障,较低优先级的代理会主动避让较高优先级的代理。
  3. 动态导航
    • 当场景中的障碍物或目标位置发生变化时,NavMeshAgent 会自动重新计算路径。
    • 例如,如果要在游戏运行时改变目标位置,可以调用 agent.SetDestination(newDestination); ,其中 newDestination 是一个 Vector3 类型的新位置。

开始实践

window→package manager 选择 Unity Registry 然后搜索 AI Navigation

在这里插入图片描述

window→ai→navigation(obsolete) ,这里是因为我所使用的Unity版本那个AI Navigation static 已经弃用了,我更改了一种使用方式。

在这里插入图片描述

选择Environment,然后为点击Inspector中的Static,接着将Environment的对象及其子对象都设置为Navigation Static.

在这里插入图片描述

点击Bake→bake

在这里插入图片描述

为Enemy添加NavMeshAgent,让敌人可以在NavMesh上面行走。

在这里插入图片描述

绘制几个路径点,用于敌人的自动巡逻。

在这里插入图片描述

在这里插入图片描述

引用巡逻点,编辑EnemyBehavior脚本

   // 巡逻点的transform
   public Transform PatrolRoute;

   // 四个巡逻点的transform集合
   public List<Transform> Locations;

   void Start()
   {
       // 脚本运行时,初始化巡逻路径
       InitializePatrolRoute();
   }

   void InitializePatrolRoute()
   {
       foreach (Transform t in PatrolRoute)
       {
           Locations.Add(t);
       }
   }

在这里插入图片描述

让Enemy跟着巡逻点动起来,在此编辑脚本

// ... 
// 默认第一个索引为0,即先走第一个巡逻点
private int _locationIndex = 0;

// NavMeshAgent的引用
private NavMeshAgent _agent;

void Start()
{
    // 获取Enemy对象上的NavMeshAgent组件的引用
    _agent = GetComponent<NavMeshAgent>();  
    // 脚本运行时,初始化巡逻路径
    InitializePatrolRoute(); 
    MoveToNextPatrolLocation();
}

// 移动到下一个巡逻点
void MoveToNextPatrolLocation()
{
    _agent.destination = Locations[_locationIndex].position;
}
//....

在这里插入图片描述

现在只会移动到第一个巡逻点,然后我们的目的是,在每个巡逻点,循环往复的进行移动,这个时候就需要在Update()中搞点事情了。

void Update()
{
    // 检测当前对象与目标位置距离
    if(_agent.remainingDistance < 0.2f && !_agent.pathPending)
    {
        MoveToNextPatrolLocation();
    }    
}
    
     // 移动到下一个巡逻点
 void MoveToNextPatrolLocation()
 {
     if(Locations.Count == 0) return;
     _agent.destination = Locations[_locationIndex].position;
     // 索引循环往复
     _locationIndex = (_locationIndex + 1) % Locations.Count;    
 }

在这里插入图片描述

🎃 敌人游戏机制

敌人按照巡逻路线进行巡逻,当与玩家碰撞后,减少玩家生命值,然后敌人回到初始位置,玩家也可以攻击敌人,减少其生命值。

追踪玩家

当玩家进入敌人警戒范围时,敌人会直接奔向玩家。

 // 玩家的Transform引用
 public Transform Player;

 void Start()
 {
     // ...
     // 获取Player对象的引用
     Player = GameObject.Find("Player").transform;
 }

 // ...

 // 碰撞触发器
 void OnTriggerEnter(Collider other)
 {
     if(other.name == "Player")
     {
		     // 玩家进入到警戒范围后,敌人奔向玩家
         _agent.destination = Player.position;
         Debug.Log("玩家进入警戒范围");
     }    
 }

攻击玩家

攻击玩家的逻辑很简单,当玩家与敌人发生碰撞时,玩家减少生命值,并且在GUI上显示

修改玩家的Playerbehavior脚本(敌人脚本也可以,因为碰撞是相互的)

// gameManager的引用,因为HP,收集道具数量都在该处存放
private GameBehavior _gameManager;

 void Start()
 {
     // 将当前player的刚体组件获取并设置
     _rb = GetComponent<Rigidbody>();
     // 获取当前角色的胶囊碰撞体
     _col = GetComponent<CapsuleCollider>();

     _gameManager = GameObject.Find("Game_Manager").
         GetComponent<GameBehavior>();
     
 }
 
 
  // 发生碰撞时
  void OnCollisionEnter(Collision collision)
  {
      // 如果与敌人碰撞则HP-1
      if (collision.gameObject.name == "Enemy")
      {
          _gameManager.HP -= 1;
      }
  }

在这里插入图片描述

这里的话需要注意一个问题,就是敌人身上的碰撞体为碰撞触发器,勾选了Is Trigger,然后直接通过Player的OnCollisionEnter()并不会触发,需要给敌人再加入一个碰撞体就可以了。

在这里插入图片描述

子弹碰撞

为Player添加反击手段,逻辑就是敌人有三滴血,当子弹射击三次并且击中敌人后,敌人自动销毁。

编辑敌人Enemy的游戏脚本

// 敌人的HP=3
private int _lives = 3;
public int EnemyLives
{
    get { return _lives; }
    set
    {
        _lives = value;

        if (_lives <= 0)
        {
            Destroy(this.gameObject);
            Debug.Log("敌人死亡");
        }
    }
}

void OnCollisionEnter(Collision collision)
{
		// 当子弹碰撞到敌人时
    if(collision.gameObject.name == "Bullet(Clone)")
    {
        EnemyLives -= 1;
        Debug.Log("遭受子弹射击");
    }
}

在这里插入图片描述

完善游戏失败条件

逻辑是,生命值为0时,游戏暂停,出现一个按钮说:你输了。

先绘制一个Button按钮,和之前的一样,默认禁用即可,当生命值为0时,在脚本内调用。

在这里插入图片描述

编辑GameManager的脚本

public Button LossButton;
private int _playerHp = 10;
public int HP
{
    get { return _playerHp; }
    set
    {
        //...
        if (_playerHp <= 0)
        {
            ProgressText.text = "You want another life with that?";
            LossButton.gameObject.SetActive(true);
            Time.timeScale = 0f;
        }
        else
        {
            ProgressText.text = "Ouch... that's got hurt.";
        }
        Debug.LogFormat("HP:{0}", _playerHp);
    }
}

在这里插入图片描述

🤗 总结归纳

本文主要介绍了在 Unity 中实现 AI 导航以及构建敌人游戏机制的方法。包括导航系统的基本组件、导航行为和逻辑,以及敌人的巡逻、追踪玩家、攻击玩家、被子弹击中和完善游戏失败条件等功能。

重要亮点

  • 导航系统基本组件:Unity 中 AI 导航的基础是 NavMesh,它将场景中的可行走表面进行三角剖分形成网格。NavMeshAgent 是附加在游戏对象上用于在 NavMesh 上移动的组件,NavMeshObstacle 用于表示动态障碍物。
  • 导航行为和逻辑:NavMeshAgent 能自动规划从当前位置到目标位置的最短路径,处理避障和动态导航。例如,当场景中的障碍物或目标位置变化时,会自动重新计算路径。
  • 敌人巡逻机制:通过为敌人添加 NavMeshAgent,设置巡逻点,编写脚本实现敌人在巡逻点之间循环往复移动。
  • 追踪玩家:当玩家进入敌人警戒范围时,敌人会奔向玩家。通过在敌人脚本中检测玩家进入碰撞触发器,设置敌人的目标位置为玩家位置。
  • 攻击玩家:当玩家与敌人发生碰撞时,玩家减少生命值。在玩家脚本和敌人脚本中分别处理碰撞事件。
  • 完善游戏失败条件:当玩家生命值为 0 时,游戏暂停,出现 “你输了” 按钮。在游戏管理器脚本中根据玩家生命值控制按钮的显示和游戏时间的暂停。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

UDP 广播组播点播的区别及联系

1、网络IP地址的分类 组播地址是分类编址的IPv4地址中的D类地址&#xff0c;又叫多播地址&#xff0c;他的前四位必须是1110&#xff0c;所以网络地址的二进制取值范围是11100000~11101111对应的十进制为 224~~239。所以以224~239开头的网络地址都是组播地址。 组播地址的功能…

css粘性定位超出指定宽度失效问题

展示效果 解决办法&#xff1a;外层容器添加display:grid即可 完整代码 <template><div class"box"><div class"line" v-for"items in 10"><div class"item" v-for"item in 8">drgg</div>&…

携程旅行 登录分析

声明: 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;抓包内容、敏感网址、数据接口等均已做脱敏处理&#xff0c;严禁用于商业用途和非法用途&#xff0c;否则由此产生的一切后果均与作者无关&#xff01; 逆向分析 所有加密流程基本一样就说…

(Java版本)基于JAVA的网络通讯系统设计与实现-毕业设计

源码 论文 下载地址&#xff1a; ​​​​c​​​​​​c基于JAVA的网络通讯系统设计与实现(源码系统论文&#xff09;https://download.csdn.net/download/weixin_39682092/90299782https://download.csdn.net/download/weixin_39682092/90299782 第1章 绪论 1.1 课题选择的…

PageView组件的功能和用法

文章目录 1 概念介绍2 使用方法3 示例代码 我们在上一章回中介绍了如何屏蔽事件关的内容,本章回中将介绍PageView Widget.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1 概念介绍 我们在这里介绍的PageView是指左右滑动或者上下滑动显示不同的页面&#xff0c;Flutter把它…

《Memory Barriers a Hardware View for Software Hackers》阅读笔记

CPU 设计者引入内存屏障&#xff08;memory barriers&#xff09;是为了应对在多处理器系统&#xff08;SMP&#xff09;中&#xff0c;内存引用重排序可能导致的同步问题。尽管重排序可以提高性能&#xff0c;但在某些情况下&#xff08;如同步原语&#xff09;&#xff0c;正…

网络编程-网络原理HTTP1

文章目录 HTTP请求/响应的基本结构认识URLURL是什么和基本格式关于encoding机制 认识方法(method)GET方法简介GET方法的特点POST方法简介POST方法的特点GET和POST的区别(经典面试题)关于GET和POST的补充说明Restful风格 上节主要是对http协议的一些最基本的概念做出一些说明, 然…

Facebook广告点击率CTR太低 如何优化

在投放Facebook广告的过程中&#xff0c;点击率CTR是一个衡量广告效果的重要指标&#xff0c;和广告质量、受众定位准确性、转化率等息息相关&#xff0c;因此&#xff0c;提升CTR是一个重要的广告优化流程。 一、什么是Facebook广告点击率&#xff0c;如何查看&#xff1f; C…

编写、应用中断例程

实验内容、程序清单及运行结果 编写、应用中断例程&#xff08;课本实验13&#xff09; 编写并安装int 7ch中断例程&#xff0c;功能为显示一个用0结束的字符串&#xff0c;中断例程安装在0:200处。 assume cs:code data segment db welcome to masm!,0 data ends code s…

WPF基础 | WPF 基础概念全解析:布局、控件与事件

WPF基础 | WPF 基础概念全解析&#xff1a;布局、控件与事件 一、前言二、WPF 布局系统2.1 布局的重要性与基本原理2.2 常见布局面板2.3 布局的测量与排列过程 三、WPF 控件3.1 控件概述与分类3.2 常见控件的属性、方法与事件3.3 自定义控件 四、WPF 事件4.1 路由事件概述4.2 事…

ssh密钥登录GitHub时一直提示“Error: Permission denied (publickey)”

起因 环境&#xff1a;Windows10 背景&#xff1a;之前就是按照官方说明创建个rsa密钥&#xff0c;在git后台添加上&#xff0c;就行了&#xff0c;近期怎么添加怎么失败&#xff0c;总是“Error: Permission denied (publickey)”的提示&#xff01; 尝试 各种尝试&#xf…

ESP32S3基于espidf lvgl驱动i2c ssd1306/sh1106/7屏幕使用

ESP32S3基于espidf lvgl驱动i2c ssd1306/sh1106/7屏幕使用 &#x1f516;个人使用的espidf版本&#xff1a;V5.4&#xff0c;lvgl组件版本&#xff1a;8.3.0&#xff08;可在idf_component.yml文件中看到&#xff09; 对于 i2c ssd1306接口屏幕&#xff0c;可以直接使用自带的d…

C++ 复习总结记录十

C 复习总结记录十 主要内容 1、stack 介绍和使用 2、queue 介绍和使用 3、priority_queue 介绍和使用 4、容器适配器 一 stack 的介绍和使用 stack 文档介绍 1、 stack 是容器适配器&#xff0c;专用于后进先出的操作&#xff0c;只能从容器尾端进行元素插入和提取 2、…

ESP32服务器和PC客户端的Wi-Fi通信

ESP32客户端-服务器Wi-Fi通信 本指南将向您展示如何设置ESP32板作为服务端&#xff0c;PC作为客户端&#xff0c;通过HTTP通信&#xff0c;以通过Wi-Fi&#xff08;无需路由器或互联网连接&#xff09;交换数据。简而言之&#xff0c;您将学习如何使用HTTP请求将一个板的数据发…

激光雷达和相机早期融合

通过外参和内参的标定将激光雷达的点云投影到图像上。 • 传感器标定 首先需要对激光雷达和相机&#xff08;用于获取 2D 图像&#xff09;进行外参和内参标定。这是为了确定激光雷达坐标系和相机坐标系之间的转换关系&#xff0c;包括旋转和平移。通常采用棋盘格等标定工具&…

机器学习-核函数(Kernel Function)

核函数&#xff08;Kernel Function&#xff09;是一种数学函数&#xff0c;主要用于将数据映射到一个更高维的特征空间&#xff0c;以便于在这个新特征空间中更容易找到数据的结构或模式。核函数的主要作用是在不需要显式计算高维特征空间的情况下&#xff0c;通过内积操作来实…

【基于无线电的数据通信链】Link 11 仿真测试

〇、废话 Link 11 仿真测试 涉及多个方面&#xff0c;包括信号仿真、协议模拟、数据链路层的仿真以及网络性能评估等。Link 11 是一种基于 HF&#xff08;高频&#xff09; 或 UHF&#xff08;超高频&#xff09; 波段的无线通信协议&#xff0c;主要用于军事通信系统中。为了…

计算机图形学:实验四 带纹理的OBJ文件读取和显示

一、程序功能设计 在程序中读取带纹理的obj文件&#xff0c;载入相应的纹理图片文件&#xff0c;将带纹理的模型显示在程序窗口中。实现带纹理的OBJ文件读取与显示功能&#xff0c;具体设计如下&#xff1a; OBJ文件解析与数据存储 通过实现TriMesh类中的readObj函数&#x…

【PVE】Proxmox VE8.0+创建LXC容器安装docker

为了不影响PVE宿主机&#xff0c;通常使用套娃的形式安装Docker容器&#xff0c;再安装相关docker应用。首先在CT模板中创建 Linux 容器&#xff0c;推荐使用Debian。开启ssh登录&#xff0c;修改debian配置&#xff0c;安装docker 一、创建 LXC 容器 1、CT模板下载 点击“模…

如何为64位LabVIEW配置正确的驱动程序

在安装 64位 LabVIEW 后&#xff0c;确保驱动程序正确配置是关键。如果您首先安装了 32位 LabVIEW 和相关驱动&#xff0c;然后安装了 64位 LabVIEW&#xff0c;需要确保为 64位 LabVIEW 安装和配置适当的驱动程序&#xff0c;才能正常访问硬件设备。以下是详细步骤&#xff1a…