Unity2D:自动寻路插件A* Pathfinding教程——实际运用与编程

最终效果

在这里插入图片描述

先上Graph图和配置

在这里插入图片描述

在这里插入图片描述

注意Diameter是用来检测是否可以通过的直接

需要考虑AI对象的碰撞器大小,以免测试通过但是卡住过不去。

这是AI对象的组件

在这里插入图片描述

向鼠标单击右键的地方移动

由于使用的是移动而不是施加力,所以可能出现超过的情况

  • 第一种处理办法是修改移动向量的长度使得当好落在目标点。但是这种办法会导致移动过程速度时快时慢,镜头移动不流畅
  • 第二种办法是循环检测下一个目标点是否在当前移动后,已经越过了

可能被卡住,所以写了个随机方向移动(移动后重新寻路)

在这里插入图片描述

杂揉了更新动画的代码,不能拿来直接用,就参考一下吧

    /// <summary>
    /// 鼠标的世界坐标
    /// </summary>
    /// <returns></returns>
    public static Vector2 MousePosInWorld()
    {
        return Camera.main.ScreenToWorldPoint(Input.mousePosition);
    }
    /// <summary>
    /// 随机方向向量
    /// </summary>
    /// <returns></returns>
    public Vector2 RandomDirection()
    {
        float x = Random.Range(-1f, 1f);
        float y = Mathf.Sqrt(1 - x * x);
        if (Random.Range(0f, 1f) > 0.5f)
        {
            y *= -1;
        }
        return new Vector2(x, y);
    }
/* 
 *  Author : Jk_Chen
 */

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Pathfinding;

public class PlayerAI : MonoBehaviour
{
    PlayerControler playerControler;

    float MaxSpeed = 10; // 最大速度
    float SpeedLimit = 0.01f; // 更新动画器的速度阈值
    Vector2 target; // 目标位置

    public bool enable; // 是否AI行动
    Path path; // 路径
    int nextP = 0; // 当前要去的点下标

    Seeker seeker;
    Rigidbody2D _rigidbody;

    void Start()
    {
        enable = false;
        playerControler = GetComponent<PlayerControler>();
        _rigidbody = GetComponent<Rigidbody2D>();
        seeker = GetComponent<Seeker>();
    }

    void FixedUpdate()
    {
        // 右键
        if (Input.GetMouseButtonDown(1))
        {
            EndAI("新的寻路");
            Vector2 vector = Helper.MousePosInWorld();
            StartAI(vector);
        }
        // AI移动
        if (enable)
        {
            // 卡住了往随机方向撤 timeLen 秒
            if (resTime > 0)
            {
                resTime -= Time.deltaTime;
                MoveRandom();
                // 结束后重新确实路径
                if (resTime <= 0)
                {
                    SeekPath();
                }
            }
            else
            {
                // 每隔20帧检查是否被卡住
                if (Time.frameCount % 20 == 0)
                    prePos = transform.position;
                MoveAI();
                if (Time.frameCount % 20 == 19)
                {
                    if (Vector2.Distance(prePos, transform.position) < 0.01f)
                    {
                        randomV = Helper.instance.RandomDirection();
                        Debug.Log(prePos + " " + transform.position);
                        resTime = timeLen;
                    }
                }
            }
        }
    }

    Vector2 prePos = new Vector2(-1e9f, -1e9f);
    float resTime;
    Vector2 randomV;
    float timeLen = 0.5f;

    /// <summary>
    /// 向下一个点移动
    /// </summary>
    void MoveAI()
    {
        if (path == null || !enable)
        {
            return;
        }
        // 获取向量
        Vector2 vector = (path.vectorPath[nextP] - transform.position);
        if (vector.magnitude > SpeedLimit)
        {
            Vector2 delta = Time.deltaTime * vector.normalized * MaxSpeed;
            playerControler.UpdateAnimator(vector.normalized);
            Vector2 pos = (Vector2)transform.position + delta;
            _rigidbody.MovePosition(pos);
            //由于使用的是移动而不是施加力,所以可能出现超过的情况
            //第一种处理办法是修改移动向量的长度使得当好落在目标点
            //      但是这种办法会导致移动过程速度时快时慢,镜头移动不流畅
            // 第二种办法是循环检测下一个目标点是否在当前移动后,已经越过了
            while (delta.magnitude > Vector2.Distance(transform.position, path.vectorPath[nextP]))
            {
                nextP++;
                if (nextP == path.vectorPath.Count)
                {
                    EndAI("已到达终点");
                    break;
                }
            }
        }
        else
        {
            nextP++;
            if(nextP == path.vectorPath.Count)
            {
                EndAI("已到达终点");
            }
            playerControler.UpdateAnimator(vector);
        }
    }

    /// <summary>
    /// 向随机生成的方向移动
    /// </summary>
    void MoveRandom()
    {
        Vector2 delta = Time.deltaTime * randomV * MaxSpeed;
        //playerControler.UpdateAnimator(delta.normalized);
        Vector2 pos = (Vector2)transform.position + delta;
        _rigidbody.MovePosition(pos);
    }

    /// <summary>
    /// 开启AI模式
    /// </summary>
    /// <param name="pos"></param>
    public void StartAI(Vector2 pos)
    {
        enable = true;
        target = pos;
        SeekPath();
    }

    /// <summary>
    /// 结束AI
    /// </summary>
    public void EndAI(string info)
    {
        enable = false;
        path = null;
        Debug.Log(info);
    }

    /// <summary>
    /// 向目标寻路
    /// </summary>
    /// <param name="pos"></param>
    void SeekPath()
    {
        if (path != null && !seeker.IsDone())
        {
            return;
        }
        Debug.Log("Try " + transform.position + " to " + target);
        seeker.StartPath(transform.position, target, OnPathComplete);
        //if (seeker.IsDone())
        //{
        //    EndAI("暂时无法到达终点");
        //}
    }

    /// <summary>
    /// 完成寻路
    /// </summary>
    /// <param name="p"></param>
    void OnPathComplete(Path p)
    {
        if (!p.error)
        {
            Debug.Log("Success Seek");
            path = p;
            nextP = 0;
        }
        else
        {
            EndAI("");
        }
    }
}


去掉Debug

在这里插入图片描述

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
烘培桥是指将场景中的网格图(NavMesh)生成并保存在磁盘上,以便在运行时进行快速的路径计算。要在Unity中使用A*路径查找算法,需要使用Unity的NavMesh系统,并使用A*寻路插件来执行路径计算。 以下是在Unity中使用A*路径查找算法的步骤: 1. 安装A*寻路插件:从Unity Asset Store中下载并导入A* Pathfinding Project Pro插件。 2. 创建场景:在Unity中创建一个场景,并添加游戏对象和地形。 3. 创建NavMesh:使用Unity的NavMesh系统来创建场景的NavMesh。在场景中选择菜单栏的“Window”->“Navigation”,打开“Navigation”窗口,然后单击“Bake”按钮来烘焙NavMesh。 4. 添加A*路径查找组件:在场景中选择要使用A*路径查找算法的游戏对象,然后右键单击并选择“Add Component”->“A* Pathfinding”->“A* Pathfinding Component”。这将添加一个A*路径查找组件到游戏对象上。 5. 设置路径查找组件:在A*路径查找组件的属性窗口中,设置起点和终点,以及其他路径查找参数,例如搜索半径、移动速度和路径平滑等。 6. 执行路径查找:在游戏运行时,调用A*路径查找组件的路径查找函数来执行路径查找。例如,在脚本中调用“GetComponent<AstarPath>().Scan()”函数来执行路径查找。 7. 显示路径:在游戏运行时,使用A*路径查找组件的路径可视化功能来显示路径。例如,在脚本中调用“GetComponent<AstarPath>().DrawPath()”函数来显示路径。 以上是在Unity中使用A*路径查找算法的基本步骤。通过使用A*寻路插件Unity的NavMesh系统,可以轻松地实现高效的路径查找功能,从而为游戏开发带来更好的用户体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值