Unity中的群组行为

一:效果演示

不使用群组算法和使用群组算法的演示视频



二:什么是群组行为

群组行为属于人工智能思想,比如说人群走路、鸟群飞行、鱼群游动,群组中成员之间不应该很规律的运动,例如不能拥挤到一起,而是更加真实的去模拟现实中的场景
下面是不使用群组行为和使用群组行为的演示

群组中的三种行为:
——分隔:成员间保持一定距离,不能太近
——队列:群组以相同的速度向相同方向运动
——聚集:避免与临近成员距离太远


三:群组行为中的三种行为

为了更好的模拟现实环境,使用施加力的方式替代直接修改物体速度,因为直接修改物体的速度会无视各种外力,而施加力是一种仿物理的方式

——分隔

separation diagram

当成员之间相距很近时,给其一个相反的力进行分离

 

——队列

alignment diagram

当与大部队运动方向不一致时,给其一个力进行调整

 

——聚集

cohesion diagram


当与其他成员相距很远时,就选取就近几个成员的中心点,施加力实现聚集


四:代码实现

using UnityEngine;

/// <summary>
/// 群组行为中的个体
/// </summary>
public class UnityFlock : MonoBehaviour
{
    public Transform target;//目标物体

    [Header("运动速度")]
    public float speed;
    [Header("物体质量")]
    private int m = 1;

    [Header("需要分离的距离")]
    [Header("====================")]
    public float separationDist;
    [Header("分离力的比重")]
    public float separationWeight;

    [Header("需要队列的距离")]
    public float alignmentDist;
    [Header("队列力的比重")]
    public float alignmentWeight;

    [Header("需要聚集的距离")]
    public float cohesionDist;
    [Header("聚集力的比重")]
    public float cohesionWeight;

    private Vector3 velocity;//速度

    private void Awake()
    {
        velocity = Vector3.forward;
    }

    private void Update()
    {
        Vector3 sumForce = Vector3.zero;
        sumForce += Separation() * separationWeight;
        sumForce += Alignment() * alignmentWeight;
        sumForce += Cohesion() * cohesionWeight;
        sumForce += Seek() * speed;

        Vector3 a = sumForce / m;
        velocity += a * Time.deltaTime;

        transform.rotation = Quaternion.Slerp(transform.rotation, Quaternion.LookRotation(velocity), 0.2f);
        transform.Translate(transform.forward * Time.deltaTime, Space.World);
    }

    /// <summary>
    /// 寻找目标
    /// </summary>
    /// <returns></returns>
    private Vector3 Seek()
    {
        Vector3 force = Vector3.zero;
        Vector3 dir = target.position - transform.position;
        force = dir.normalized - transform.forward;
        return force;
    }

    /// <summary>
    /// 得到分离的力
    /// </summary>
    private Vector3 Separation()
    {
        Vector3 separationForce = Vector3.zero;
        GameObject[] objs = GameObject.FindGameObjectsWithTag("UnityFlock");
        foreach (var unity in objs)
        {
            if (unity == gameObject)
            {
                continue;
            }
            if (Vector3.Distance(transform.position, unity.transform.position) <= separationDist)
            {
                Vector3 dir = transform.position - unity.transform.position;
                separationForce += dir.normalized / dir.magnitude;
            }
        }
        return separationForce;
    }

    /// <summary>
    /// 得到队列的力
    /// </summary>
    private Vector3 Alignment()
    {
        Vector3 alignmentForce = Vector3.zero;
        GameObject[] objs = GameObject.FindGameObjectsWithTag("UnityFlock");
        Vector3 avgDir = Vector3.zero;
        int counter = 0;
        foreach (var unity in objs)
        {
            if (unity == gameObject)
            {
                continue;
            }
            if (Vector3.Distance(transform.position, unity.transform.position) < alignmentDist)
            {
                counter++;
                avgDir += unity.transform.forward;
            }
        }
        avgDir /= counter;
        alignmentForce += (avgDir - transform.forward).normalized;
        return alignmentForce;
    }

    /// <summary>
    /// 聚集
    /// </summary>
    private Vector3 Cohesion()
    {
        Vector3 cohesionForce = Vector3.zero;
        GameObject[] objs = GameObject.FindGameObjectsWithTag("UnityFlock");
        Vector3 avgDir = Vector3.zero;
        int counter = 0;
        foreach (var unity in objs)
        {
            if (unity == gameObject)
            {
                continue;
            }
            if (Vector3.Distance(transform.position, unity.transform.position) < alignmentDist)
            {
                counter++;
                avgDir += unity.transform.position;
            }
        }
        avgDir /= counter;
        cohesionForce += (avgDir - transform.position).normalized;
        return cohesionForce;
    }
}

 

原文地址:https://blog.csdn.net/LLLLL__/article/details/115803620

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


这篇文章主要介绍了Unity游戏开发中外观模式是什么意思,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家...
这篇文章主要介绍Unity中地面检测方案的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!1.普通射线在角色坐标(一般是脚底)...
这篇文章主要介绍了Unity游戏开发中如何消除不想要的黄色警告,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带...
这篇文章主要介绍了Unity中有多少种渲染队列,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解
这篇文章主要介绍Unity中如何实现Texture,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!了解Texture2D 如上图,Texture2D是一张
小编给大家分享一下Unity中DOTS要实现的特点有哪些,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让...
这篇文章给大家分享的是有关unity中如何实现UGUI遮罩流光特效的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。下面是核心shader:Sh...
这篇文章主要为大家展示了“Unity中如何实现3D坐标转换UGUI坐标”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下...
这篇文章主要介绍了Unity游戏开发中设计模式的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家...
这篇文章主要介绍了Unity中如何实现仿真丝袜渲染,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了...
这篇文章给大家分享的是有关Unity插件OVRLipSync有什么用的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。项目需要接入对话口型动...
这篇文章主要介绍了Unity性能优化之DrawCall的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家...
这篇文章给大家分享的是有关Unity给力插件之Final IK怎么用的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。这插件有什么用:一般游...
这篇文章给大家分享的是有关Unity中如何内嵌网页插件UniWebView的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。一、常见Unity中内...
小编给大家分享一下Unity如何做流体物理的几个轮子,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让...
小编给大家分享一下Unity中Lod和Occlusion Culling的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收...
这篇文章将为大家详细讲解有关Unity中LineRenderer与TrailRenderer有什么用,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获...
这篇文章主要介绍了Unity中coroutine问题的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起...
这篇文章将为大家详细讲解有关unity中spine怎么用,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。骨骼动画首先我们来看到...
这篇文章主要为大家展示了“Unity Shader后处理中如何实现简单均值模糊”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学...