问小白 wenxiaobai
资讯
历史
科技
环境与自然
成长
游戏
财经
文学与艺术
美食
健康
家居
文化
情感
汽车
三农
军事
旅行
运动
教育
生活
星座命理

Unity DOTS技术深度解析:ECS、Job System与Burst编译器

创作时间:
作者:
@小白创作中心

Unity DOTS技术深度解析:ECS、Job System与Burst编译器

引用
CSDN
1.
https://m.blog.csdn.net/yupu56/article/details/146099754

Unity DOTS(Data-Oriented Technology Stack)是Unity推出的一种用于开发高性能游戏和应用的数据导向技术栈。它通过ECS架构、Job System和Burst编译器三大核心组件,为开发者提供了强大的性能优化工具。本文将详细介绍DOTS的核心概念、优势及其具体实现方式。

什么是Dots

DOTS(Data-Oriented Technology Stack)是Unity推出的一种用于开发高性能游戏和应用的数据导向技术栈,包含三大核心组件:

  • ECS架构:实体(Entity)+ 组件(Component)+ 系统(System)的分离式设计
  • Job System:提供在多个 CPU 核心上并行运行游戏代码的方法
  • Burst编译器:将C#代码编译为优化的机器码

这些使得 Unity 能够生成高度优化的代码。

DOTS的优势

1.性能优化:传统的面向对象编程模型可能会导致性能瓶颈,尤其是在需要处理大量相似实体(如敌人、项目、粒子等)时。DOTS 通过将数据放在内存中以连续的方式存储并通过实体组件系统(ECS)进行处理,显著提高了内存访问效率和 CPU 利用率。

2.提高并行性:DOTS 中的 Job System 允许开发者轻松地将任务并行化,这样可以充分利用现代CPU多核处理器的优势。通过将游戏逻辑拆分成多个独立的任务,能够同时在多个线程中执行,极大地提高了计算性能。

2.可扩展性:随着游戏复杂性的增加,管理和处理大量实体的需求也随之提高。DOTS 的设计理念使其在处理大规模数据时更加高效,能够轻松支持成百上千的实体而不影响性能。

4.组件化设计:ECS 的组件化模型通过将数据与逻辑分开,提供了更灵活和可重用的代码结构。开发者可以独立修改和组合不同的组件,简化维护和扩展的过程。

5.优化内存使用:DOTS 鼓励以数据驱动的方式组织代码,更少依赖于复杂的对象引用。这样做有助于减少内存碎片,提高内存的整体利用效率。

示例:传统MonoBehaviour在10,000+对象场景下帧率可能低于30FPS,DOTS可实现60FPS以上
硬件利用率提升:多核CPU并行处理效率提升300%+(测试数据:i7-12700K处理20,000实体)

通常被称为DOTS的三个支柱,可作为单独的Unity软件包使用。它们共同构成了交付面向数据的高性能解决方案的基础。

ECS(实体组件系统)

ECS 从面向对象的方法转变为面向数据的方法使得重用代码变得更加容易,也使得其他人更容易理解和使用它。它是一个与 GameObjects 兼容的面向数据的框架,凭借前所未有的控制和确定性水平,使经验丰富的 Unity 创作者能够实现更多。

实体创建流程:

// 传统MonoBehaviour转换示例
public class UnitMoverAuthoring : MonoBehaviour
{
    public float moveSpeed = 2;
    public float rotationSpeed = 2;
    
    //Baker 类负责将 GameObject 的数据和组件转化为可以在 DOTS 中操作的 ECS 实体。
    private class Baker : Baker<UnitMoverAuthoring>
    {
        public override void Bake(UnitMoverAuthoring authoring)
        {
            Entity entity = GetEntity(TransformUsageFlags.Dynamic);
            AddComponent(entity, new UnitMover
            {
                moveSpeed = authoring.moveSpeed,
                rotationSpeed = authoring.rotationSpeed,
                targetPosition = float3.zero,
            });
        }
    }
}
//ComponnetData 数据类
public struct UnitMover : IComponentData
{
    public float moveSpeed;
    public float rotationSpeed;
    public float3 targetPosition;
}
// System 系统逻辑
public partial struct MoveSystem : ISystem {
    public void OnUpdate(ref SystemState state) {
        foreach (var (transform, speed) in 
                SystemAPI.Query<RefRW<LocalTransform>, RefRO<MoveSpeed>>()) {
            transform.ValueRW.Position.y += speed.ValueRO.Value * SystemAPI.Time.DeltaTime;
        }
    }
}

Job System作业系统

C# 作业系统

该系统使 Unity 开发人员能够利用多核计算平台,并行化代码可以安全、高速地运行。C# 作业系统暴露了 Unity 内部的 C++ 作业系统,使 Unity 创作者能够在 Unity 内部处理的同时运行他们的脚本。

//我们定义一个组件,表示实体的速度。
[GenerateAuthoringComponent]
public struct MoveSpeed : IComponentData
{
    public float Value;
}
//创建 JobEntity
public struct MoveJob : IJobEntity
{
    public float DeltaTime; // 用于控制移动的增量
    // Execute 方法会对每个具有 MoveSpeed 组件的实体调用
    public void Execute(ref MoveSpeed moveSpeed, ref Translation translation)
    {
        translation.Value += moveSpeed.Value * DeltaTime; // 移动逻辑
    }
}
//在系统中使用 JobEntity
public class MoveSystem : SystemBase
{
    protected override void OnUpdate()
    {
        // 获取每帧的时间增量
        float deltaTime = Time.DeltaTime;
        // 创建一个 MoveJob 实例
        var moveJob = new MoveJob 
        {
            DeltaTime = deltaTime // 设置时间增量
        };
        // 调度 Job,处理所有包含 MoveSpeed 和 Translation 组件的实体
        this.Dependency = moveJob.ScheduleParallel(this.Dependency);
    }
}

Burst编译器

Burst是一种编译器,可将IL/.NET字节码转换为高度优化的本地代码。它使用业界公认的 LLVM 编译器基础架构,为游戏创作者提供 C# 原生代码的性能。Burst 还暴露了 CPU 的内在特性,从而可以对性能关键代码进行微调。

最后

Unity DOTS技术栈为开发者提供了强大的性能优化工具,通过ECS架构、Job System和Burst编译器三大核心组件,可以显著提升游戏性能和开发效率。对于希望开发高性能游戏的Unity开发者来说,DOTS是一个值得深入学习的重要技术方向。

© 2023 北京元石科技有限公司 ◎ 京公网安备 11010802042949号