UNITY实战进阶-BatchRendererGroup+Jobs+Burst+RVO2+GPUAnimation 实现万人团战

本文主要是介绍UNITY实战进阶-BatchRendererGroup+Jobs+Burst+RVO2+GPUAnimation 实现万人团战,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

研究思路:GPUAnimation把动画放入GPU中处理,BatchRendererGroup进行动态批量渲染处理,Jobs+Burst进行多线程处理逻辑(移动、攻击等),RVO2采用Jobs的寻路导航。

准备工作:
Editor => Project Setting =>Package Manager => Enable Pre-release Package勾选
PackageManager安装以下:Jobs、Burs、Universal RP


URP渲染管线

Rendering=>Create “Universal Render Pipeline Asset”
Editor=>Project Setting=>Quality=>RenderPipelineAsset=>“Universal Render Pipeline Asset”
Editor=>Project Setting=>Graphics=>ScriptsableRenderPipelineSetting=>“Universal Render Pipeline Asset”


Job System C#任务系统

  1. 在Job System对外之前,Unity虽然内部是多线程处理,但是外部的代码都必须跑在主线程上。
  2. C#虽然支持Thread,但是在Unlty中只能处理数据,例如:网络消息、下载。如果想在Thread中调用Unity的API那是不行的。
  3. 有了Job System就可以充分利用CPU的多核,例如:在多线程中修改Transform旋转、缩放、平移
  4. 例如:MMO游戏判断碰撞、大量同步角色坐标、大量的血条飘字等都比较适合Job System
  5. Unity没有直接将Thread开放出来,可以有效避免Thread被滥用,开发者可放心使用Job而不用太多关心如线程安全、加锁这些问题。
  6. Job最好配合Burst编译器,这样能生成高效的本地代码(后面会细聊Burst)

job中只能使用值类型(float,int,uint,short,bool。。。),enums,structs和其他类型的指针。
job中不能使用引用类型,T[]数组属于引用类型,无法使用,使用NativeArray<T>[]替代。

所以:接口IJob是个开线程的操作,再job中按顺序执行,所以可以准确的保证值的准确性

    //在job中按顺序执行,所以可以准确的保证值的准确性struct MyStruct : IJob{public NativeArray<float> values;public float offset;public void Execute(){for (int i = 0; i < values.Length; i++){values[i] += offset;}}}void Update(){NativeArray<float> values = new NativeArray<float>(500, Allocator.Persistent);MyStruct myStruct = new MyStruct();myStruct.values = values;myStruct.offset = 10;for (int i = 0; i < 1000; i++){JobHandle jobHandle = myStruct.Schedule();jobHandle.Complete();}values.Dispose();}

接口IJobParallelFor,在job中按并行执行,内部会自动加锁

    //在job中按并行执行,内部会自动加锁struct MyStruct2 : IJobParallelFor{//声明数据是只读的,意味着copy这个数据不需要加锁[ReadOnly]public NativeArray<float> copy;//非声明ReadOnly,默认values为write/read,数据一旦改变,job会等待public NativeArray<float> values;public void Execute(int index){values[index] = copy[index];}}void Update(){NativeArray<float> values = new NativeArray<float>(500, Allocator.Persistent);NativeArray<float> copy = new NativeArray<float>(500, Allocator.Persistent);for (int i = 0; i < copy.Length; i++){copy[i] = Random.Range(0, int.MaxValue);}MyStruct2 myStruct = new MyStruct2();myStruct.values = values;myStruct.copy = copy;for (int i = 0; i < 1000; i++){JobHandle jobHandle = myStruct.Schedule(values.Length, 32);jobHandle.Complete();}values.Dispose();copy.Dispose();}


Burs编译器

  1. Burst编译器是以LLVM为基础的后端编译技术。
  2. 编译器的原理会分5个步骤;源代码—》前端—》优化器—》后端—》机器码。
  3. LLVM的定义了个抽象语言IR,前端负责将源代码(C#)编译成IR,优化器负责优化IR,后端负责将IR生成目标语言这里就是机器码。
  4. 正是因为抽象语言IR的存在,所以LLVM支持的语言很多,而且也方便扩展C#、ActionScript、Ada、D语言、Fortran、GLSL、Haskell、Java字节码、Objective-C、Swift、Python、Ruby、Rust、Scala等 语言。
  5. LLVM代码是开源了,所以Unlty很适合用它来做Burst的编译。
  6. 遗憾的是LLVM对C#的GC做的不好,所以burst只支持值类型数据编译,不支持引用类型数据编译

Unity.Mathematics数学库

  1. Unity.Mathematics提供矢量类型(float4,float3..)它可以直接映射到硬件SIMD寄存器。
  2. Unlty.Mathematics的Math类中也提供了直接映射到硬件SIMD寄存器。
  3. 这样原本CPU需要一个个计算的,有了SIMD可以一次性计算完毕。
  4. 需要注意的是unlty之前的Math类默认是不支持映射SIMD寄存器的。

        如何启动BursCompile,我们可以在C#代码中使用Burst编译器。在需要使用Burst编译器的方法或类上添加[BurstCompile]属性。

        我们需要在Unity3D项目的Player Settings中启用Burst编译器。在Unity3D编辑器中,选择Edit -> Project Settings -> Player,在Inspector窗口中找到Scripting Backend选项,选择IL2CPP,并勾选“Enable Burst Compilation”选项即可。

  1. Struct上添加BursCompile即可。
  2. Struct必须是继承IJob类,否则无效。
 //在job中按顺序执行,所以可以准确的保证值的准确性[BurstCompile]struct MyStruct : IJob{public NativeArray<float> values;public float offset;public void Execute(){for (int i = 0; i < values.Length; i++){values[i] += offset;}}}

设计模式-组合模式: 类似GameObject上绑定组件,组件化。


未完待续

这篇关于UNITY实战进阶-BatchRendererGroup+Jobs+Burst+RVO2+GPUAnimation 实现万人团战的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/875289

相关文章

Nginx实现动态封禁IP的步骤指南

《Nginx实现动态封禁IP的步骤指南》在日常的生产环境中,网站可能会遭遇恶意请求、DDoS攻击或其他有害的访问行为,为了应对这些情况,动态封禁IP是一项十分重要的安全策略,本篇博客将介绍如何通过NG... 目录1、简述2、实现方式3、使用 fail2ban 动态封禁3.1 安装 fail2ban3.2 配

Java中实现订单超时自动取消功能(最新推荐)

《Java中实现订单超时自动取消功能(最新推荐)》本文介绍了Java中实现订单超时自动取消功能的几种方法,包括定时任务、JDK延迟队列、Redis过期监听、Redisson分布式延迟队列、Rocket... 目录1、定时任务2、JDK延迟队列 DelayQueue(1)定义实现Delayed接口的实体类 (

将java程序打包成可执行文件的实现方式

《将java程序打包成可执行文件的实现方式》本文介绍了将Java程序打包成可执行文件的三种方法:手动打包(将编译后的代码及JRE运行环境一起打包),使用第三方打包工具(如Launch4j)和JDK自带... 目录1.问题提出2.如何将Java程序打包成可执行文件2.1将编译后的代码及jre运行环境一起打包2

Java使用Tesseract-OCR实战教程

《Java使用Tesseract-OCR实战教程》本文介绍了如何在Java中使用Tesseract-OCR进行文本提取,包括Tesseract-OCR的安装、中文训练库的配置、依赖库的引入以及具体的代... 目录Java使用Tesseract-OCRTesseract-OCR安装配置中文训练库引入依赖代码实

通俗易懂的Java常见限流算法具体实现

《通俗易懂的Java常见限流算法具体实现》:本文主要介绍Java常见限流算法具体实现的相关资料,包括漏桶算法、令牌桶算法、Nginx限流和Redis+Lua限流的实现原理和具体步骤,并比较了它们的... 目录一、漏桶算法1.漏桶算法的思想和原理2.具体实现二、令牌桶算法1.令牌桶算法流程:2.具体实现2.1

MySQL8.0设置redo缓存大小的实现

《MySQL8.0设置redo缓存大小的实现》本文主要在MySQL8.0.30及之后版本中使用innodb_redo_log_capacity参数在线更改redo缓存文件大小,下面就来介绍一下,具有一... mysql 8.0.30及之后版本可以使用innodb_redo_log_capacity参数来更改

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在