Dynamic Buffers
A DynamicBuffer
is a type of component data that allows a variable-sized, "stretchy"
buffer to be associated with an entity. It behaves as a component type that
carries an internal capacity of a certain number of elements, but can allocate
a heap memory block if the internal capacity is exhausted.
Memory management is fully automatic when using this approach. Memory associated with
DynamicBuffers
is managed by the EntityManager
so that when a DynamicBuffer
component is removed, any associated heap memory is automatically freed as well.
DynamicBuffers
supersede fixed array support which has been removed.
Declaring Buffer Element Types
To declare a Buffer
, you declare it with the type of element that you will be
putting into the Buffer
:
// This describes the number of buffer elements that should be reserved
// in chunk data for each instance of a buffer. In this case, 8 integers
// will be reserved (32 bytes) along with the size of the buffer header
// (currently 16 bytes on 64-bit targets)
[InternalBufferCapacity(8)]
public struct MyBufferElement : IBufferElementData
{
// These implicit conversions are optional, but can help reduce typing.
public static implicit operator int(MyBufferElement e) { return e.Value; }
public static implicit operator MyBufferElement(int e) { return new MyBufferElement { Value = e }; }
// Actual value each buffer element will store.
public int Value;
}
While it seem strange to describe the element type and not the Buffer
itself,
this design enables two key benefits in the ECS:
It supports having more than one
DynamicBuffer
of typefloat3
, or any other common value type. You can add any number ofBuffers
that leverage the same value types, as long as the elements are uniquely wrapped in a top-level struct.We can include
Buffer
element types inEntityArchetypes
, and it generally will behave like having a component.
Adding Buffer Types To Entities
To add a buffer to an entity, you can use the normal methods of adding a component type onto an entity:
Using AddBuffer()
entityManager.AddBuffer<MyBufferElement>(entity);
Using an archetype
Entity e = entityManager.CreateEntity(typeof(MyBufferElement));
Accessing Buffers
There are several ways to access DynamicBuffers
, which parallel access methods
to regular component data.
Direct, main-thread only access
DynamicBuffer<MyBufferElement> buffer = entityManager.GetBuffer<MyBufferElement>(entity);
Entity based access
You can also look up Buffers
on a per-entity basis from a JobComponentSystem:
var lookup = GetBufferFromEntity<EcsIntElement>();
var buffer = lookup[myEntity];
buffer.Append(17);
buffer.RemoveAt(0);
Reinterpreting Buffers (experimental)
Buffers
can be reinterpreted as a type of the same size. The intention is to
allow controlled type-punning and to get rid of the wrapper element types when
they get in the way. To reinterpret, simply call Reinterpret<T>
:
var intBuffer = entityManager.GetBuffer<EcsIntElement>().Reinterpret<int>();
The reinterpreted Buffer
carries with it the safety handle of the original
Buffer
, and is safe to use. They use the same underlying BufferHeader
, so
modifications to one reinterpreted Buffer
will be immediately reflected in
others.
Note that there are no type checks involved, so it is entirely possible to
alias a uint
and float
buffer.