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

Unity DOTS从入门到精通之 C# Job System

文章目录

    • 前言
    • 安装 DOTS 包
    • C# 任务系统
      • Mono 环境
      • DOTS 环境
      • 运行作业
      • NativeContainer

前言

作为 DOTS 教程,我们将创建一个旋转立方体的简单程序,并将传统的 Unity 设计转换为 DOTS 设计。

  • Unity 2022.3.52f1
  • Entities 1.3.10

安装 DOTS 包

要安装 DOTS 包,请按照以下步骤操作:

(1)从菜单“窗口 → 包管理器”打开包管理器。
(2)搜索“ Entities” 并安装 Entities和Entities Graphics。
(3)搜索“ Universal RP” 并安装 Universal RP,并设置Graphics/Scriptable Render Pipeline Settings。

这会添加“实体”作为依赖项,并递归添加它所依赖的关联包( Burst、Collections、Jobs、Mathematics等)。

在这里插入图片描述

C# 任务系统

“ C# 作业系统”是用于执行并行处理的功能。您只需执行作业即可充分利用 CPU 内核,而不必担心执行的顺序或时间。

其特点包括:
・代码简洁
・无GC
・安全
・快速

当“主线程”上无法执行所有处理时,将创建一个“作业”,将处理划分为多个较小的进程,并将这些进程添加(调度)到“作业队列”中。 “工作线程”从“作业队列”中取出一个“作业”并执行。
在这里插入图片描述
此时,C# 作业系统管理依赖关系,以便按适当的顺序执行作业。例如,如果 JobB 依赖于 JobA,那么您可以确保在 JobA 完成之前 JobB 不会运行。

虽然Job可以在Mono和Dots环境下运行,但是Mono下只能使用多线程的基本能力。
我们这里推荐在Dots环境下使用JobSystem

Mono 环境

仅依赖 JobSystem 的多线程能力

// 需继承 MonoBehaviour
public class MonoJobSample : MonoBehaviour {
    void Update() {
        var job = new MyJob { 
        	/* 数据传递 */ 
        };
        JobHandle handle = job.Schedule();
        handle.Complete(); // 需手动同步
    }
}

DOTS 环境

Dots环境下配合Burst,才真正能够发挥JobSystem的并行能力
但是Job的写法也很重要,写法不规范依然不能充分使用Burst的能力。

  • job内联写法,Lambda写法,适合简单组件遍历
public void OnUpdate(ref SystemState state) {
    Entities.ForEach((ref LocalTransform trans, ref FindTarget find) => {
        // 自动处理多线程调度
    }).ScheduleParallel();
}
  • IJobEntity 标准实现,适合复杂逻辑,支持Burst编译
public partial struct HealthSystem : ISystem
{
    private EntityCommandBuffer.ParallelWriter _ecb;
    
    [BurstCompile]
    public void OnUpdate(ref SystemState state)
    {
        _ecb = SystemAPI.GetSingleton<EndSimulationEntityCommandBufferSystem.Singleton>()
            .CreateCommandBuffer(state.WorldUnmanaged).AsParallelWriter();
        HealthJob unitMoverJob = new HealthJob
        {
            ECB = _ecb,
        };
        unitMoverJob.ScheduleParallel();
    }
}

[BurstCompile]
public partial struct HealthJob : IJobEntity
{
    public EntityCommandBuffer.ParallelWriter ECB;

    public void Execute([EntityIndexInQuery] int index,
        ref Health health,in PostTransformMatrix localTransform,
        Entity entity)
    {
		//这里做System逻辑处理
		//逻辑是跑在Burst多线程中,性能提升明显
    }
}

运行作业

1.定义一个作业
要定义一个作业,首先准备一个继承“ IJob ”的结构。接下来,准备在字段中作业处理中要使用的变量。最后用Execute()实现 Job 的处理。

2.创建 Job
“Job”是作为继承“ IJob ” 的结构体( struct )生成的。

Job 字段可以有两种类型:
・基础类型:int、float、bool 等。
・NativeContainer:NativeArray、NativeSlice、NativeList 等。

只有“NativeContainer”数据可以在 Job 和主线程之间共享;“原始类型”可用于输入但不能用于输出。

3.指定Job的执行方式
指定Job的执行方式有三种方式:

・Run():在主线程中执行 lambda 表达式。在这种情况下,等待作业完成。
・Schedule():将 lambda 表达式安排为单个作业。
・ScheduleParallel():将 lambda 表达式安排为分成多个块的作业。

返回的值是“ JobHandle ”。这是对计划作业进行操作的句柄。

4.JobHandle 操作:
通过调用“JobHandle”方法上的 Complete() 等待 Job 完成。

NativeContainer

“ NativeContainer ” 是一个非托管容器,不同于 C# 提供的托管容器(List、Dictionary 等)。由于它不受 GC 管理,因此您需要通过调用Dispose()自行释放内存。

其特点包括:

  • 自己决定内存分配类型(分配器)
  • 使用后必须使用 Dispose() 释放内存
  • 仅限结构(不允许使用类)
  • 不能增加元素的数量

“NativeContainer”的类型有:

・NativeArray<Value>:数组
・NativeSlice<Value>:从 NativeArray 中切出一部分
・NativeList<Value>:列表
・NativeHashMap<Key, Value>:字典
・NativeMultiHashMap<Key, Value>:每个键有多个值的字典
・NativeQueue<Value>:先进先出(FIFO)队列

“C#作业系统”使用“NativeContainer”实现作业和主线程之间的数据共享。用于在Job和主线程之间传递数据。

创建 NativeContainer
创建一个“NativeArray”,即 NativeContainer 之一。第一个参数是“元素的数量”,第二个参数是“内存分配类型”。

内存分配类型有:

・Allocator.Temp:用于分配和释放一帧或更少的内存。
・Allocator.TempJob:4帧内使用的内存分配和释放。
Allocator.Persistent :应用程序生命周期内的持久分配。

相关文章:

  • LeeCode题库第四十六题
  • 长上下文 GRPO
  • Core Speech Kit(基础语音服务)
  • Django与视图
  • 大整数加法(信息学奥赛一本通-1168)
  • 锂电池组的串数设计研究
  • QOJ9700 Ying’s Cup(拉格朗日插值优化卷积,背包,二项式反演)
  • c++ 嵌入汇编的方式实现int型自增
  • Javascript ajax
  • Compose笔记(八)--权限
  • 《Operating System Concepts》阅读笔记:p208-p227
  • 更新vscode ,将c++11更新到c++20
  • springboot旅游管理系统设计与实现(代码+数据库+LW)
  • openharmory-鸿蒙生态设备之间文件互传发现、接入认证和文件传输
  • 通义千问本地配置并实现微调
  • 手写Tomcat:实现基本功能
  • FastAPI Cookie 和 Header 参数完全指南:从基础到高级实战 [特殊字符]
  • MyBatis-Plus 注解大全
  • Java字符串拼接的底层原理与性能优化
  • C++类和对象入门(三)
  • 营销类wordpress主题/windows优化大师要会员
  • 杭州网站改版公司电话/人工智能培训师
  • 滨江区住房和城乡建设局网站/郑州网站优化顾问
  • 做汽配网站需要多少钱/专业seo网站
  • 怎么更换网站logo/广告公司注册
  • 腾讯云建设网站教程/百度联盟怎么加入赚钱