Unity DOTS中的Archetype与Chunk

Unity DOTS中的Archetype与Chunk

在Unity中,archetype(原型)用来表示一个world里具有相同component类型组合的entity。也就是说,相同component类型的entity在Unity内部会存储到一起,共享同一个archetype。

Unity DOTS中的Archetype与Chunk1

使用这样的设计,可以高效地通过component类型进行查询。例如,如果想查找所有具有component类型 A 和 B 的entity,那么只需找到表示这两个component的所有archetype,这比扫描所有entity更高效。

entity和componenet并不直接存储在archetype中,而是存储在称为chunk的连续内存块中。每个块由16KB组成,它们实际可以存储的entity数量取决于chunk所属archetype中component的数量和大小。 这样做可以更好匹配缓存行,同时也能更好支持并行操作。

在这里插入图片描述

对于一个chunk,它除了包含用于每种component类型的数组之外,还包含一个额外的数组来存储entity的ID。例如,在具有component类型 A 和 B 的archetype中,每个chunk都有三个数组:一个用于component A的数组,一个用于component B的数组,以及一个用于entity ID的数组。

Unity DOTS中的Archetype与Chunk3

chunk中的数据是紧密连续的,chunk的每个entity都存储在每个数组连续的索引中。添加一个新的entity时,会将其存储在第一个可用索引中。而当从chunk中移除entity时,chunk的最后一个entity将被移动以填补空隙。添加entity时,如果对应archetype的所有chunk都已满,则Unity将创建一个全新的chunk。类似地,当从chunk中移除最后一个entity时,Unity会将该chunk进行销毁。

在编辑器中,Unity提供了Archetypes窗口,可以方便预览当前所有的archetype。我们先用代码创建4个entity,然后给它们分别添加两种component:

public partial struct FirstSystem : ISystem
{public void OnCreate(ref SystemState state) { Entity e1 = state.EntityManager.CreateEntity();Entity e2 = state.EntityManager.CreateEntity();Entity e3 = state.EntityManager.CreateEntity();Entity e4 = state.EntityManager.CreateEntity();state.EntityManager.AddComponentData(e1, new FirstComponent { Value = 1 });state.EntityManager.AddComponentData(e2, new FirstComponent { Value = 2 });state.EntityManager.AddComponentData(e3, new FirstComponent { Value = 3 });state.EntityManager.AddComponentData(e3, new SecondComponent { Value = 13 });state.EntityManager.AddComponentData(e4, new FirstComponent { Value = 4 });state.EntityManager.AddComponentData(e4, new SecondComponent { Value = 14 });}
}public struct FirstComponent : IComponentData
{public int Value;
}public struct SecondComponent : IComponentData
{public int Value;
}

那么,根据之前的分析,e1和e2应该属于同一个archetype,而e3和e4则属于另一个archetype。在编辑器中运行,观察Archetypes窗口,可以发现的确如此:

Unity DOTS中的Archetype与Chunk4

Unity DOTS中的Archetype与Chunk5

根据官网的解释,Archetypes窗口中可以观察到以下信息:

PropertyDescription
Archetype nameThe archetype name is its hash, which you can use to find the archetype again across future Unity sessions.
EntitiesNumber of entities within the selected archetype.
Unused EntitiesThe total number of entities that can fit into all available chunks for the selected Archetype, minus the number of active entities (represented by the entities stat).
ChunksNumber of chunks this archetype uses.
Chunk CapacityThe number of entities with this archetype that can fit into a chunk. This number is equal to the total number of Entities and Unused Entities.
ComponentsDisplays the total number of components in the archetype and the total amount of memory assigned to them in KB. To see the list of components and their individual memory allocation, expand this section.
External ComponentsLists the Chunk components and Shared components that affect this archetype.

自此,我们对Unity的archetype和chunk有了基本的认识。接下来我们来看看源代码,深入一下细节部分。

在上面的例子中,我们使用EntityManager.CreateEntity这个函数来创建entity。函数内部会在创建entity之前,检查是否存在相应的archetype。由于此时尚不存在表示一个不含任何component类型的空entity的archetype,那么函数会先去创建archetype:

internal EntityArchetype CreateArchetype(ComponentType* types, int count, bool addSimulateComponentIfMissing)
{Assert.IsTrue(types != null || count == 0);EntityComponentStore->AssertCanCreateArchetype(types, count);ComponentTypeInArchetype* typesInArchetype = stackalloc ComponentTypeInArchetype[count + 2];var cachedComponentCount = FillSortedArchetypeArray(typesInArchetype, types, count, addSimulateComponentIfMissing);return CreateArchetype_Sorted(typesInArchetype, cachedComponentCount);
}

每个空的entity都要额外添加一个Simulate类型的component,这也是上面的截图中为何会多出一个component的原因。Unity首先会尝试从已有的空闲archetypes中寻找符合条件的archetype,没找到才会去真正创建一个。

public Archetype* GetOrCreateArchetype(ComponentTypeInArchetype* inTypesSorted, int count)
{var srcArchetype = GetExistingArchetype(inTypesSorted, count);if (srcArchetype != null)return srcArchetype;srcArchetype = CreateArchetype(inTypesSorted, count);return srcArchetype;
}

Unity的Archetype是一个非常复杂的struct,它记录了所代表的component信息,以及chunk信息,还有它们当前的状态:

[StructLayout(LayoutKind.Sequential)]
internal unsafe struct Archetype
{public ArchetypeChunkData Chunks;public UnsafeList<ChunkIndex> ChunksWithEmptySlots;public ChunkListMap FreeChunksBySharedComponents;public ComponentTypeInArchetype* Types; // Array with TypeCount elementspublic int* EnableableTypeIndexInArchetype; // Array with EnableableTypesCount elements// back pointer to EntityQueryData(s), used for chunk list cachingpublic UnsafeList<IntPtr> MatchingQueryData;// single-linked list used for invalidating chunk list cachespublic Archetype* NextChangedArchetype;public int EntityCount;public int ChunkCapacity;public int TypesCount;public int EnableableTypesCount;public int InstanceSize;public int InstanceSizeWithOverhead;public int ScalarEntityPatchCount;public int BufferEntityPatchCount;public ulong StableHash;public ulong BloomFilterMask;// The order that per-component-type data is stored in memory within an archetype does not necessarily match// the order that types are stored in the Types/Offsets/SizeOfs/etc. arrays. The memory order of types is stable across// runs; the Types array is sorted by TypeIndex, and the TypeIndex <-> ComponentType mapping is *not* guaranteed to// be stable across runs.// These two arrays each have TypeCount elements, and are used to convert between these two orderings.// - MemoryOrderIndex is the order an archetype's component data is actually stored in memory. This is stable across runs.// - IndexInArchetype is the order that types appear in the archetype->Types[] array. This is *not* necessarily stable across runs.//   (also called IndexInTypeArray in some APIs)public int* TypeMemoryOrderIndexToIndexInArchetype; // The Nth element is the IndexInArchetype of the type with MemoryOrderIndex=Npublic int* TypeIndexInArchetypeToMemoryOrderIndex; // The Nth element is the MemoryOrderIndex of the type with IndexInArchetype=N// These arrays each have TypeCount elements, ordered by IndexInArchetype (the same order as the Types array)public int*    Offsets; // Byte offset of each component type's data within this archetype's chunk buffer.public ushort* SizeOfs; // Size in bytes of each component typepublic int*    BufferCapacities; // For IBufferElementData components, the buffer capacity of each component. Not meaningful for non-buffer components.// Order of components in the types array is always:// Entity, native component data, buffer components, managed component data, tag component, shared components, chunk componentspublic short FirstBufferComponent;public short FirstManagedComponent;public short FirstTagComponent;public short FirstSharedComponent;public short FirstChunkComponent;public ArchetypeFlags Flags;public Archetype* CopyArchetype; // Removes cleanup componentspublic Archetype* InstantiateArchetype; // Removes cleanup components & prefabspublic Archetype* CleanupResidueArchetype;public Archetype* MetaChunkArchetype;public EntityRemapUtility.EntityPatchInfo* ScalarEntityPatches;public EntityRemapUtility.BufferEntityPatchInfo* BufferEntityPatches;// @macton Temporarily store back reference to EntityComponentStore.// - In order to remove this we need to sever the connection to ManagedChangesTracker//   when structural changes occur.public EntityComponentStore* EntityComponentStore;public fixed byte QueryMaskArray[128];
}

其中,TypesCount表示archetype所保存的数据种类,一般就是entity+component类型数量,这里空entity自带一个Simulate的component,那么TypesCount即为2;Types就是对应的类型数组了;EntityCount为当前archetype所拥有的entity数量;ChunksWithEmptySlots表示archetype中所有空闲chunk的索引列表,archetype就是通过它来为entity指定chunk的;然后,一系列First打头的Component(FirstBufferComponentFirstManagedComponentFirstTagComponentFirstSharedComponentFirstChunkComponent)表示在type数组中第一个出现该类型的索引;SizeOfs是一个比较重要的字段,它表示每个component type所占的内存大小,通过它可以进而计算出archetype最多可拥有entity的数量,也就是ChunkCapacity字段;在此基础上,还可以计算出每个component type在chunk中的起始位置Offsets;最后,Chunks是一个ArchetypeChunkData类型的字段,它用来记录archetype中所有chunk的状态。

有了archetype之后,就可以进入创建entity的阶段了,那么下一步,Unity会先查找是否有可用的chunk,没有的话就得真正分配内存,去创建一个:

ChunkIndex GetChunkWithEmptySlots(ref ArchetypeChunkFilter archetypeChunkFilter)
{var archetype = archetypeChunkFilter.Archetype;fixed(int* sharedComponentValues = archetypeChunkFilter.SharedComponentValues){var chunk = archetype->GetExistingChunkWithEmptySlots(sharedComponentValues);if (chunk == ChunkIndex.Null)chunk = GetCleanChunk(archetype, sharedComponentValues);return chunk;}
}

注意到这里函数返回的是ChunkIndex类型,它不是真正意义上的chunk,而是对chunk的二次封装,用来方便快捷地存取chunk的数据,以及记录chunk的各种状态与属性。ChunkIndex只有一个int类型的value变量,表示当前chunk在所有chunk中的位置。虽然Unity声称每个chunk只有16KB大小,但是实际分配内存时,并不是按16KB进行分配,而是会一口气先分配好1 << 20字节也就是64 * 16KB大小的内存,这块内存被称为mega chunk。所有mega chunk的内存指针都保存在大小为16384个ulong类型的连续内存中,因此ChunkIndex记录的是全局索引,第一步先索引到某个mega chunk,然后再定位到真正的chunk地址。

在这里插入图片描述

internal Chunk* GetChunkPointer(int chunkIndex)
{var megachunkIndex = chunkIndex >> log2ChunksPerMegachunk;var chunkInMegachunk = chunkIndex & (chunksPerMegachunk-1);var megachunk = (byte*)m_megachunk.ElementAt(megachunkIndex);var chunk = megachunk + (chunkInMegachunk << Log2ChunkSizeInBytesRoundedUpToPow2);return (Chunk*)chunk;
}

Chunk是一个相对简单的数据结构,前64字节为chunk头部,后面才是数据内容。

[StructLayout(LayoutKind.Explicit)]
internal unsafe struct Chunk
{// Chunk header START// The following field is only used during serialization and won't contain valid data at runtime.// This is part of a larger refactor, and this field will eventually be removed from this struct.[FieldOffset(0)]public int ArchetypeIndexForSerialization;// 4-byte padding to keep the file format compatible until further changes to the header.[FieldOffset(8)]public Entity metaChunkEntity;// The following field is only used during serialization and won't contain valid data at runtime.// This is part of a larger refactor, and this field will eventually be removed from this struct.[FieldOffset(16)]public int CountForSerialization;[FieldOffset(28)]public int ListWithEmptySlotsIndex;// Special chunk behaviors[FieldOffset(32)]public uint Flags;// SequenceNumber is a unique number for each chunk, across all worlds. (Chunk* is not guranteed unique, in particular because chunk allocations are pooled)[FieldOffset(kSerializedHeaderSize)]public ulong SequenceNumber;// NOTE: SequenceNumber is not part of the serialized header.//       It is cleared on write to disk, it is a global in memory sequence ID used for comparing chunks.public const int kSerializedHeaderSize = 40;// Chunk header END// Component data buffer// This is where the actual chunk data starts.// It's declared like this so we can skip the header part of the chunk and just get to the data.public const int kBufferOffset = 64; // (must be cache line aligned)[FieldOffset(kBufferOffset)]public fixed byte Buffer[4];public const int kChunkSize = 16 * 1024;public const int kBufferSize = kChunkSize - kBufferOffset;public const int kMaximumEntitiesPerChunk = kBufferSize / 8;public const int kChunkBufferSize = kChunkSize - kBufferOffset;
}

随后,Unity会在该chunk中为要创建的entity分配索引,索引为当前chunk的entity数量,因为之前提到chunk的数据都是连续排布的,中间并不存在空隙。

static int AllocateIntoChunk(Archetype* archetype, ChunkIndex chunk, int count, out int outIndex)
{outIndex = chunk.Count;var allocatedCount = Math.Min(archetype->ChunkCapacity - outIndex, count);SetChunkCount(archetype, chunk, outIndex + allocatedCount);archetype->EntityCount += allocatedCount;return allocatedCount;
}

根据索引,就可以将chunk中属于该entity的部分(entity id,component type)进行初始化,clear内存等操作,自此,一个entity就算创建完成了。

for (var t = 1; t != typesCount; t++)
{var offset = offsets[t];var sizeOf = sizeOfs[t];var dst = dstBuffer + (offset + sizeOf * dstIndex);if (types[t].IsBuffer){for (var i = 0; i < count; ++i){BufferHeader.Initialize((BufferHeader*)dst, bufferCapacities[t]);dst += sizeOf;}}else{UnsafeUtility.MemClear(dst, sizeOf * count);}
}

接下来,再为entity添加component,那么原来entity所在的chunk和archetype不再满足要求,需要将entity移动到新的chunk和archetype中,这种行为在Unity中被称作structural change。在AddComponent之前,Unity首先会检查entity是否已有该component,同一类型的component只允许存在一个。这个时候archetype就发挥作用了,只需要去archetype中查找component type是否存在即可:

public bool HasComponent(Entity entity, ComponentType type)
{if (Hint.Unlikely(!Exists(entity)))return false;var archetype = GetArchetype(entity);return ChunkDataUtility.GetIndexInTypeArray(archetype, type.TypeIndex) != -1;
}

如果entity没有该component,那么就需要找一下符合条件的archetype以及空闲的chunk。

ChunkIndex GetChunkWithEmptySlotsWithAddedComponent(ChunkIndex srcChunk, ComponentType componentType, int sharedComponentIndex = 0)
{var archetypeChunkFilter = GetArchetypeChunkFilterWithAddedComponent(srcChunk, componentType, sharedComponentIndex);if (archetypeChunkFilter.Archetype == null)return ChunkIndex.Null;return GetChunkWithEmptySlots(ref archetypeChunkFilter);
}

新的archetype和chunk都有了,剩下的就是move操作了,先把数据从源chunk拷贝到目标chunk中去,更新目标archetype,然后再清理源chunk,更新源archetype。

int Move(in EntityBatchInChunk srcBatch, ChunkIndex dstChunk)
{var srcChunk = srcBatch.Chunk;var srcArchetype = GetArchetype(srcChunk);var dstArchetype = GetArchetype(dstChunk);var dstUnusedCount = dstArchetype->ChunkCapacity - dstChunk.Count;var srcCount = math.min(dstUnusedCount, srcBatch.Count);var srcStartIndex = srcBatch.StartIndex + srcBatch.Count - srcCount;var partialSrcBatch = new EntityBatchInChunk{Chunk = srcChunk,StartIndex = srcStartIndex,Count = srcCount};ChunkDataUtility.Clone(srcArchetype, partialSrcBatch, dstArchetype, dstChunk);fixed (EntityComponentStore* store = &this){ChunkDataUtility.Remove(store, partialSrcBatch);}return srcCount;
}

前面提到过,当从chunk中移除entity时,chunk的最后一个entity将被移动以填补空隙。其实就是个简单的拷贝内存操作:

var fillStartIndex = chunk.Count - fillCount;Copy(entityComponentStore, chunk, fillStartIndex, chunk, startIndex, fillCount);

Reference

[1] Archetypes concepts

[2] 深入理解ECS:Archetype(原型)

[3] Archetypes window reference

[4] Structural changes concepts

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

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

相关文章

一篇文章入门Pytest!

目录 一、安装 二、语法 三、执行 四、前后置setup/teardown 五、fixture 六、数据驱动 七、报告 一、安装 需要安装的库 pytest pytest-html 生成HTML格式的测试报告 pytest-xdist 用例分布式执行&#xff0c;多CPU分发 pytest-ordering …

【Bug】iOS 不支持运行或调试你的项目的上一个生成版本。 请先确保生成解决方案,再运行或调试它。

文章目录 问题问题代码原因解决处理Bug的具体步骤 问题 在windows以hot restart&#xff08;hot restart不需要mac 而pair to mac需要&#xff09;的方式部署到iphone报&#xff1a;不支持运行或调试你的项目的上一个生成版本。 请先确保生成解决方案&#xff0c;再运行或调试…

基于SSM宠物信息交流平台JAVA|VUE|Springboot计算机毕业设计源代码+数据库+LW文档+开题报告+答辩稿+部署教+代码讲解

源代码数据库LW文档&#xff08;1万字以上&#xff09;开题报告答辩稿 部署教程代码讲解代码时间修改教程 一、开发工具、运行环境、开发技术 开发工具 1、操作系统&#xff1a;Window操作系统 2、开发工具&#xff1a;IntelliJ IDEA或者Eclipse 3、数据库存储&#xff1a…

东芝TB67B008FTG三相无刷马达驱动IC

TB67B008FTG是一款无刷直流电机用的三相PWM驱动器&#xff0c;专为无传感器的应用场景设计。它具有一系列突出的功能&#xff0c;使其在现代电机控制中发挥重要作用&#xff0c;尤其是在需要高效、稳定运行的场景中。本文将详细介绍TB67B008FTG的特点、应用以及它在电机控制领域…

idea和webstorm性能优化

idea和webstorm性能优化 简介 今天打开了idea弹了一个弹窗&#xff0c;大概意思如下。 The IDE has detected Microsoft Defender with Real-Time Protection enabled. It might severely degrade IDE performance. It is recommended to add the following paths to the Def…

【Axure高保真原型】移动案例

今天和大家分享多个常用的移动案例的原型模板&#xff0c;包括轮盘滑动控制元件移动、页面按钮控制元件移动、鼠标单击控制元件移动、元件跟随鼠标移动、鼠标拖动控制元件移动、键盘方向键控制元件移动&#xff0c;具体效果可以点击下方视频观看或打开下方预览地址查看哦 【原…

算法笔记day06

目录 1.添加逗号 2.跳台阶 3.扑克牌顺子 1.添加逗号 添加逗号_牛客题霸_牛客网 算法思路&#xff1a; 按照提议模拟即可&#xff0c;从后向前遍历字符串&#xff0c;遍历三个字符之后&#xff0c;将其插入将这三个字符插入到新的字符串中再加上逗号。 #include <iostrea…

Thymeleaf模板引擎教程(详细总结)

Thymeleaf 是一个服务器端 Java 模板引擎&#xff0c;能够处理 HTML 、 XML 、 CSS 、 JAVASCRIPT 等模板文件。 Thymeleaf 模板可以直接当作静态原型来使用&#xff0c;它主要目标是为开发者的开发工作流程带来优雅的自然 模板&#xff0c;也是 Java 服务器端 HTML5 开…

Docker设置日志滚动

问题描述 Docker 容器中的进程会将打印到控制台(console)的日志保存到容器的目录下&#xff0c;默认的 Docker 配置不带有日志的回滚。会在自己的容器目录下往同一个日志文件中不停写入&#xff0c;最后会导致磁盘空间占满的问题。 解决方案 方案一&#xff1a;全局范围内修…

一文掌握Cephadm部署Ceph存储集群

&#x1f4da; 博客主页&#xff1a; StevenZeng学堂 &#x1f389; 本文专栏: 一文读懂Kubernetes一文读懂Harbor云原生安全实战指南云原生存储实践指南 ❤️ 摘要&#xff1a;随着企业数据量的增长和存储需求的复杂化&#xff0c;Ceph因其高可扩展性和灵活性&#xff0c;能…

网络安全的五大误区,你中招了吗?

在数字化时代&#xff0c;网络安全问题日益突出&#xff0c;许多人在使用网络过程中存在一些误区&#xff0c;导致个人信息泄露、财产损失等问题。本文将为您揭示网络安全的五大误区&#xff0c;帮助您提高安全防范意识。 误区一&#xff1a;使用复杂密码就一定安全 许多人认为…

考研读研生存指南,注意事项

本视频课程&#xff0c;涉及考研读研的方方面面&#xff0c;从考研初试→复试面试→研究生生活→导师相处→论文专利写作混毕业&#xff0c;应有尽有。有了他&#xff0c;你的研究生生涯稳了。 读研考研注意事项&#xff0c;研究生生存指南。_哔哩哔哩_bilibili 一、考研初试注…

临阵磨枪!这份软考中级集成案例分析答题万金油赶紧收藏

在系统集成项目管理工程师案例分析科目的考试中&#xff0c;主要分为“计算题”和“分析题”两大类。 计算题主要围绕着进度管理和成本管理进行出题&#xff0c;比如挣值计算、网络图、关键路径等等&#xff0c;一般占据一道大题。 而分析题呢主要占三道大题&#xff0c;主要…

前端开发 环境变量 process.env.NODE_ENV 是什么

背景&#xff1a; 前端开发过程中&#xff0c;解决Access跨域问题&#xff0c;使用跨域代理&#xff0c;注意这里是指前端的跨域代理&#xff0c;所以这里配置的只适用于开发环境。至于生产环境一般由后端配置跨域代理&#xff0c;一般使用ngnix解决生产环境的跨域代理。 一、…

新款任天堂switch游戏机方案,支持4K60HZ投屏方案,显示器,手柄方案

据传任天堂将推出新的一代的switch掌机&#xff0c;而新款掌机将支持4K60HZ投屏 都2402年了再做1080P确实有点不太象话了 4K60HZ相较于1080P能够提升很多游戏体验&#xff0c;这时不管是HDMI显示器或者是VR眼睛清晰度都会让人舒服很多。 不过新一代的任天堂似乎也在PD协议上…

家政小程序搭建,数字化市场发展下的意义

家政服务行业作为当下社会生活中不可或缺的行业&#xff0c;需求量在逐渐增加&#xff0c;行业发展也趋向多样化。 随着数字化的浪潮&#xff0c;家政行业逐渐向数字化、智能化升级发展&#xff0c;推动行业高质量发展&#xff0c;迎合现代化发展趋势&#xff0c;这一转型为行…

网站分享丨UU在线工具

在日常的工作、学习和生活中&#xff0c;我们常常会遇到各种各样需要借助工具来解决的问题。今天就给大家介绍一个功能强大、涵盖众多实用工具的在线平台 ——UU 在线工具。 一、丰富多样的工具分类 1.文档处理类&#xff1a; PDF 工具&#xff1a;提供了 PDF 转 Word、PDF 合…

IDEA中文乱码�

这篇文章网上到处都是&#xff0c;但我写作的初衷是为了更好地审视自己的作品&#xff0c;并通过不断的总结与反思来提升自我。 文章目录 前言原因分析解决方案一、设置字体为支持中文的字体二、设置字符编码为 UTF-8三、修改 IDEA 配置文件&#xff0c;让其支持中文编码第一种…

资讯 | 财富通科技政务协同办公管理软件通过麒麟软件适配认证

2024年9月25日&#xff0c;财富通科技研发的政务协同办公管理软件成功通过中国国产操作系统麒麟软件的适配认证。本次认证是继公司区块链产品“基于区块链的企业及人员资质数字证书服务平台”认证以后得第二次认证。这一成就标志着财富通科技在推动国产软件生态建设方面迈出了坚…

Java项目-基于springboot框架的广场舞团系统项目实战(附源码+文档)

作者&#xff1a;计算机学长阿伟 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、ElementUI等&#xff0c;“文末源码”。 开发运行环境 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;SpringBoot、Vue、Mybaits Plus、ELementUI工具&#xff1a;IDEA/…