Unity C# Job System介绍(二) 安全性系统和NativeContainer

C# Job System中的安全性系统

https://docs.unity3d.com/Manual/JobSystemSafetySystem.html​docs.unity3d.com

资源竞争

当我们编写多线程代码时,经常会有产生资源竞争的风险。资源竞争会在一项操作的输出依赖于另一项它掌控之外的操作时发生。

资源竞争并不总是视为一个bug,但它却是不确定行为发生的原因之一。当资源竞争确实引发了一个bug时,因为是偶然发生的,因此很难找到问题发生的确切原因,你只能在偶然情况下才能重现这种问题。调试时问题可能就消失了,因为断点和日志可能改变单个进程的执行时机。因此资源竞争成为了编写多线程代码时最大的挑战。

安全性系统

为了让用户更容易地编写多线程代码,Unity中的C# Job System会检测所有潜在的资源竞争,从而避免用户遇到由此产生的bug

举例来说:如果C# Job System需要在主线程中发送一个数据的引用给一个Job,Job在写入对应数据的时候无法判断主线程是否也在同时操作该数据。这种情况下就会导致资源竞争。

C# Job System通过给每一个需要操作数据的Job一份数据的拷贝而不是主线程中数据的引用来避免这个问题。拷贝和原本的数据独立,从而排除了资源竞争。

C# Job System拷贝数据的方式表明了一个Job只能访问可以位块传输的数据类型(blitable data types)。这种数据类型在托管代码和原生代码之间进行传递的时候不需要类型转换。

C# Job System可以使用memcpy来拷贝可位块传输数据,并在Unity的托管部分和原生部分之间传递它们。它在安排job时使用memcpy将数据放入原生内存,并给予托管部分在job执行时访问这份拷贝数据的接口。查阅更多的信息,查看Scheduling jobs

NativeContainer

https://docs.unity3d.com/Manual/JobSystemNativeContainer.html​docs.unity3d.com

安全性系统中拷贝数据的缺点是单个job的计算结果是与外部隔离的。为了突破这个限制,我们需要把结果放在一种共享内存——NativeContainer中。

什么是NativeContainer?

NativeContainer是一种托管的数据类型,为原生内存提供一种相对安全的C#封装。它包括一个指向非托管分配内存的指针。当和Unity C# Job System一起使用时,一个NativeContainer使得一个Job可以访问和主线程共享的数据,而不是在一份拷贝数据上工作。

有什么可用的NativeContainer类型?

Unity使用一个叫做NativeArray的NativeContainer。你还可以通过一个NativeSlice来操作一个NativeArray,从而获得从某个特定位置开始确定长度的NativeArray子集。

注意:Entity Component System(ECS)包扩展了Unity.Collections命名空间,包括了其他类型的NativeArray:

  • NativeList - 一个可变长的NativeArray
  • NativeHashMap - 键值对
  • NativeMultiHashMap - 每个Key可以对应多个值
  • NativeQueue - 一个先进先出(FIFO)队列

NativeContainer和其安全性系统

安全性系统是所有NativeContainer类型的组成部分。它会追踪所有关于任何NaiveContainer的读写。

注意:所有关于NativeContainer类型的安全性检查(包括下标边界检查,内存释放检查和资源竞争检查),只在Unity Editor和Play Mode中生效。(译者:即只在编辑器环境中进行检查)

安全性系统是由DisposeSentinelAtomicSafetyHandle组成的。DisposeSentinel检测内存泄漏同时在你没有正确释放内存的时候给你一个错误信息。但内存泄漏的错误只有在泄露发生很久之后才会触发。

使用AtomicSafetyHandle在代码中进行NativeContainer所有权的转移。举例来说,如果两个已经安排的jobs向同一个NativeArray写入数据,安全性系统会抛出一个异常,带有明确的错误信息关于为什么以及如何解决这个问题。安全性系统会在你安排一个违规的job后抛出一个异常。

在这种情况下,你可以在安排job的时候添加一个依赖。第一个job可以写入到NativeContainer,一旦它执行完毕,下一个job可以安全地读取和写入同一个NativeContainer。读取和写入的限制同样影响在访问主线程中的数据时生效。安全性系统允许多个jobs并行的读取同一份数据。

通常来说,当一个job有NativeContainer的访问权限时,它同时拥有读取和写入的权限。这种配置会使性能变差。一个C# Job System不允许你在有job正在对一个NativeContainer进行读写的时候,安排另一个job对该NaiveContainer拥有写入权限。

如果某个job不需要向某个NativeContainer写入,可以将该NativeContainer加上[ReadOnly]属性,像这样

[ReadOnly]
public NativeArray<int> input;

在上面的例子中,你可以在其他jobs拥有该NativeContainer只读权限的时候同时执行该job。

注意:这边没有针对从一个job中访问静态数据的保护。访问静态数据可以绕过所有的安全性系统并可能导致Unity奔溃。关于更多的信息,可以查看C# Job System建议和错误定位

NativeContainer分配器(Allocator)

当创建一个NativeContainer时,你必须指定你需要的内存分配类型。分配的类型由jobs运行的时间来决定。这种情况下你可以在每一种情况下使分配器达到可能的最好性能。

这里对于NativeContainer的内存分配有三个分配器类型。当你初始化你的NativeContainer时你需要指定一个合适的分配器。

  • Allocator.Temp是最快的分配类型。它适用于分配一个生命周期只有一帧或更短时间的操作。你不应当把一个分配器为Temp类型分配的NativeContainer传递给jobs使用。你同时需要在函数返回之前调用Dispose方法(例如MonoBehaviour.Update,或者其他从原生到托管代码的调用)
  • Allocator.TempJob是相比于Temp是一个较慢的分配类型但它比Persistent要快。这是一个生命周期为四帧的内存分配而且它是线程安全的。如果你在四帧之内没有调用Dispose,控制台会打印一个由原生代码生成的警告信息。绝大部分小jobs使用这种类型的NativeContainer分配器。
  • Allocator.Persistent是最慢的分配类型但,它可以持续存在到你需要的时间,如果必要的话可以贯穿应用程序的整个生命周期。它是直接调用malloc的一个封装。长时间的jos可以使用这种分配类型。当性能比较紧张的时候你不应当使用Persistent

使用示例:

NativeArray<float> result = new NativeArray<float>(1, Allocator.TempJob);

注意:上例中的1表明了NativeArray的长度。在这个例子中,它只有一个数组元素(因为它只在result中存储了一块数据)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/246228.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Unity C# Job System介绍(三) Job的创建和调度

创建Jobs Unity - Manual: Creating jobs​docs.unity3d.com 为了在Unity中创建一个job你需要实现IJOb接口。IJob允许你调度一个job&#xff0c;和其他jobs并发执行。Unity - Manual: Creating jobs为了在Unity中创建一个job你需要实现IJOb接口。IJob允许你调度一个job&#…

Unity C# Job System介绍(四) 并行化Job和故障排除(完结)

并行化job ParallelFor jobs​docs.unity3d.com 当调度Jobs时&#xff0c;只能有一个job来进行一项任务。在游戏中&#xff0c;非常常见的情况是在一个庞大数量的对象上执行一个相同的操作。这里有一个独立的job类型叫做IJobParallelFor来处理此类问题。ParallelFor jobs当调…

C# Job System

概述 设计目的&#xff1a;简单安全地使用多线程&#xff0c;随便就能写出高性能代码 收益&#xff1a;FPS更高&#xff0c;电池消耗更低&#xff08;Burst编译器&#xff09; 并行性&#xff1a;C# Job System和Unity Native Job System共享工作线程worker threads&#xf…

Unity游戏开发——C#特性Attribute与自动化

这篇文章主要讲一下C#里面Attribute的使用方法及其可能的应用场景。 比如你把玩家的血量、攻击、防御等属性写到枚举里面。然后界面可能有很多地方要根据这个枚举获取属性的描述文本。 比如你做网络框架的时候&#xff0c;一个协议号对应一个类的处理或者一个方法。 比如你做…

Unity c#中Attribute用法详解

举两个例子&#xff0c;在变量上使用[SerializeFiled]属性&#xff0c;可以强制让变量进行序列化&#xff0c;可以在Unity的Editor上进行赋值。 在Class上使用[RequireComponent]属性&#xff0c;就会在Class的GameObject上自动追加所需的Component。 以下是Unity官网文档中找…

attribute的用法--C#

一直以来都没理解attribute是个什么东西&#xff0c;也没怎么用&#xff0c;但是看msdn或者git上源码使用的还是蛮频繁的&#xff0c;今天好好整理了下&#xff0c;写下自己的理解和例子&#xff1a; attribute主要用来说明代码段的的信息&#xff0c;标志等&#xff1b;可以一…

走进LWRP(Universal RP)的世界

走进LWRP&#xff08;Universal RP&#xff09;的世界 原文&#xff1a;https://connect.unity.com/p/zou-jin-lwrp-universal-rp-de-shi-jie LWRP自Unity2018发布以来&#xff0c;进入大家视野已经有一段时间了&#xff0c;不过对于广大Unity开发者来说&#xff0c;依然相对…

Unity Fine Prued Tiled Light (FPTL)

Fine Prued Tiled Light Lists 视频讲解&#xff1a;https://www.bilibili.com/video/av90517615 FPT是在Tile裁剪的基础上在进行一次剔除 可用于Forward Render和Deferred Shading两种渲染管线 利用了并行架构进行优化 将线程分为线程组再分为多个线程&#xff0c;Thread Gro…

Unity 2017 Game Optimization 读书笔记(1)Scripting Strategies Part 1

1.Obtain Components using the fastest method Unity有多种Getcomponet的方法&#xff1a; GetComponent(string), GetComponent<T>() GetComponent(typeof(T)) 哪种效率最高会跟随Unity版本的变化而变化&#xff0c;对于Unity 2017&#xff0c;本书作者的测试是Ge…

C# 多态相关的文章

一 C# 多态的实现 封装、继承、多态&#xff0c;面向对象的三大特性&#xff0c;前两项理解相对容易&#xff0c;但要理解多态&#xff0c;特别是深入的了解&#xff0c;对于初学者而言可能就会有一定困难了。我一直认为学习OO的最好方法就是结合实践&#xff0c;封装、继承在…

C++ 虚函数和虚表

几篇写的不错的文章&#xff0c;本文是整合了这几篇文章&#xff0c;感谢这些大佬 https://www.jianshu.com/p/00dc0d939119 https://www.cnblogs.com/hushpa/p/5707475.html https://www.jianshu.com/p/91227e99dfd7 多态: 多态是面相对象语言一个重要的特性,多态即让同一…

Unity 2017 Game Optimization 读书笔记(2)Scripting Strategies Part 2

1. Share calculation output 和上一个Tip很像&#xff0c;可以缓存计算结果或者各种信息&#xff0c;避免多次重复的计算&#xff0c;例如在场景里查找一个物体&#xff0c;从文件读取数据&#xff0c;解析Json等等。 容易忽略的点是常常在基类了实现了某个方法&#xff0c;在…

Unity 2017 Game Optimization 读书笔记(3)Scripting Strategies Part 3

1.Avoid retrieving string properties from GameObjects 通常来讲&#xff0c;从C#的object中获取string 属性没有额外的内存开销&#xff0c;但是从Unity中的Gameobject获取string属性不一样&#xff0c;这会产生上一篇讲到的 Native-Managed Bridge&#xff08;Native内存和…

Unity 2017 Game Optimization 读书笔记(4)Scripting Strategies Part 4

1.Avoid Find() and SendMessage() at runtime SendMessage() 方法和 GameObject.Find() 相关的一系列方法都是开销非常大的。SendMessage()函数调用的耗时大约是一个普通函数调用的2000倍&#xff0c;GameObject.Find() 则和场景的复杂度相关&#xff0c;场景越复杂&#xff0…

团队行为心理学读书笔记(1)

阿伯拉罕马斯洛需求五个层次&#xff1a; &#xff08;1&#xff09;生理需要&#xff1a;食物、水、住所、性满足等方面的需要。 &#xff08;2&#xff09;安全需要&#xff1a;保护自己免受身体和情感伤害的需要。 &#xff08;3&#xff09;爱和归属的需要&#xff1a;渴…

团队行为心理学读书笔记(2)招聘背后的心理学

招聘时&#xff0c;不要只看应聘者的知识和技能 美国著名心理学家麦克利兰1973年提出了一个著名的素质冰山模型。所谓“冰山模型”&#xff0c;就是将人员个体素质的不同表现形式划分为表面的“冰山以上部分”和深藏的“冰山以下部分”。其中&#xff0c;冰山以上部分包括基本…

团队行为心理学读书笔记(3)领导力背后的行为心理学

有小缺点的主管&#xff0c;在下属眼里更有魅力 一位心理学教授曾做了一个关于管理者魅力的实验&#xff0c;他给被测试的对象播放了四段情节类似的访谈录像&#xff1a;出现在第一段录像里的是一个非常优秀的成功人士&#xff0c;他成就辉煌&#xff0c;面对主持人的采访&…

团队行为心理学读书笔记(4)带队伍背后的行为心理学

成就高效团队的基础&#xff1a;信任 信任具有三个特征&#xff1a;第一&#xff0c;信任者承担着一定的风险&#xff0c;信任者要承受被信任者失信的损失&#xff1b;第二&#xff0c;被信任者的行为不在信任者的控制之内&#xff1b;第三&#xff0c;如果某一方违约&#xf…

Unity HDRP中的光照烘焙测试(Mixed Lighing )和间接光

部分内容摘抄自&#xff1a;https://www.cnblogs.com/murongxiaopifu/p/8553367.html 直接光和间接光 大家都知道在Unity中&#xff0c;我们可以在场景中布置方向光、点光、聚光等类型的光源。但如果只有这些光&#xff0c;则场景内只会受到直接光的影响&#xff0c;而所谓的…

聊聊Unity项目管理的那些事:Git-flow和Unity

感谢原作者https://www.cnblogs.com/murongxiaopifu/p/6086849.html 0x00 前言 目前所在的团队实行敏捷开发已经有了一段时间了。敏捷开发中重要的一个话题便是如何对项目进行恰当的版本管理。项目从最初使用svn到之后的Git One Track策略再到现在的GitFlow策略&#xff0c;中…