Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/jacksondunstan/nativecollections
Native Collection Types for Unity
https://github.com/jacksondunstan/nativecollections
csharp unity unity-scripts unity3d unity3d-development unity3d-script
Last synced: 2 days ago
JSON representation
Native Collection Types for Unity
- Host: GitHub
- URL: https://github.com/jacksondunstan/nativecollections
- Owner: jacksondunstan
- License: mit
- Created: 2018-08-06T04:44:24.000Z (over 6 years ago)
- Default Branch: master
- Last Pushed: 2023-08-26T16:58:48.000Z (over 1 year ago)
- Last Synced: 2025-01-12T16:06:28.349Z (10 days ago)
- Topics: csharp, unity, unity-scripts, unity3d, unity3d-development, unity3d-script
- Language: C#
- Homepage: https://jacksondunstan.com/articles/tag/native-collection
- Size: 231 KB
- Stars: 411
- Watchers: 16
- Forks: 32
- Open Issues: 5
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE.md
Awesome Lists containing this project
README
# Native Collections
A small library of native collections like `NativeArray` suitable to be used with Unity 2018.1 or greater. No additional packages (e.g. ECS or Burst) are required.
[![openupm](https://img.shields.io/npm/v/com.jacksondunstan.native-collections?label=openupm®istry_uri=https://package.openupm.com)](https://openupm.com/packages/com.jacksondunstan.native-collections/)
# Getting Started
Clone or download this repository and copy the `JacksonDunstanNativeCollections` directory somewhere inside your Unity project's `Assets` directory.
Add this as a package to your project by adding the below as an entry to the dependencies in the `/Package/manifest.json` file:
```json
"com.jacksondunstan.native-collections": "https://github.com/jacksondunstan/NativeCollections.git"
```For more information on adding git repositories as a package see the [Git support on Package Manager](https://docs.unity3d.com/Manual/upm-git.html) in the Unity Documentation.
# NativeLinkedList<T>
This is a doubly-linked list backed by parallel arrays. Here's how to use it:
```csharp
// Create an empty list with capacity for five nodes
NativeLinkedList list = new NativeLinkedList(5, Allocator.Temp);// Add some nodes to the list
list.InsertAfter(list.Tail, 10);
list.InsertAfter(list.Tail, 20);
list.InsertAfter(list.Tail, 30);
list.InsertAfter(list.Tail, 40);
list.InsertAfter(list.Tail, 50);// Iterate over the list
foreach (int val in list)
{
Debug.Log(val);
}// Remove a node from the middle
list.Remove(list.GetEnumeratorAtIndex(2));// Insert a node into the middle
list.InsertBefore(list.Head.Next, 15);// Sort the nodes so they can be accessed sequentially
list.SortNodeMemoryAddresses();// Access the nodes sequentially
for (int i = 0; i < list.Length; ++i)
{
Debug.Log(list[i]);
}// Dispose the list's native memory
list.Dispose();
```There is much more functionality available. See [the source](JacksonDunstanNativeCollections/NativeLinkedList.cs) for more.
To read about the making of this type, see this [article series](https://jacksondunstan.com/articles/4865).
# NativeChunkedList<T>
This is a dynamically-resizable list backed by arrays that store "chunks" of elements and an array of pointers to those chunks. Here's how to use it:
```csharp
// Create an empty list with capacity for 4096 elements in 1024 element chunks
NativeChunkedList list = new NativeChunkedList(
1024,
4096,
Allocator.Temp);// Add some elements to the list
list.Add(10);
list.Add(20);
list.Add(30);
list.Add(40);
list.Add(50);// Iterate over the list
foreach (int val in list)
{
Debug.Log(val);
}// Remove an element from the middle
list.RemoveAt(2);// Insert an element into the middle
list.Insert(1, 15);// Access the elements sequentially
foreach (var e in list.Chunks)
{
foreach (int val in e)
{
Debug.Log(val);
}
}// Dispose the list's native memory
list.Dispose();
```There is much more functionality available. See [the source](JacksonDunstanNativeCollections/NativeChunkedList.cs) for more.
To read about the making of this type, see this [article series](https://jacksondunstan.com/articles/4963).
# NativeHashSet<T>
This is a collection of unique keys that aren't mapped to values. Here's how to use it:
```csharp
// Create an empty set
NativeHashSet set = new NativeHashSet(100, Allocator.Persistent);// Add some keys
set.TryAdd(123);
set.TryAdd(456);// Check for containment
set.Contains(123); // true
set.Contains(1000); // false// Remove a key
set.Remove(123);
```There is much more functionality available. See [the source](JacksonDunstanNativeCollections/NativeHashSet.cs) for more.
To read about the making of this type, see this [article](https://jacksondunstan.com/articles/5346).
# NativeArray2D<T>
This is a 2D version of `NativeArray`. Here's how to use it:
```csharp
// Create a 2x3 empty array
NativeArray2D array = new NativeArray2D(2, 3, Allocator.Temp);// Set elements of the array
array[0, 1] = 123;
array[1, 2] = 456;// Get elements of the array
int val123 = array[0, 1];
int val456 = array[1, 2];// Iterate over the array
foreach (int val in array)
{
Debug.Log(val);
}// Copy to a managed array
int[,] managed = new int[2, 3];
array.CopyTo(managed);
```There is much more functionality available. See [the source](JacksonDunstanNativeCollections/NativeArray2D.cs) for more.
To read about the making of this type, see this [article](https://jacksondunstan.com/articles/5416).
# NativeIntPtr and NativeLongPtr
These are pointers to a single `int` or `long`, useful for counters among other purposes. Here's how to use `NativeIntPtr` (`NativeLongPtr` is identical):
```csharp
// Construct with the zero value
NativeIntPtr intPtr0 = new NativeIntPtr(Allocator.Temp);// Construct with a custom value
NativeIntPtr intPtr = new NativeIntPtr(Allocator.Temp, 123);// Read and write the value
intPtr.Value = 20;
Debug.Log("Value: " + intPtr.Value); // prints "Value: 20"// Get a Parallel for use in an IJobParallelFor
NativeIntPtr.Parallel parallel = intPtr.GetParallel();// Perform atomic writes on it
parallel.Increment();
parallel.Decrement();
parallel.Add(100);// Dispose the native memory
intPtr0.Dispose();
intPtr.Dispose();
```To read about the making of this type, see this [article](https://jacksondunstan.com/articles/4940).
# NativePerJobThreadIntPtr and NativePerJobThreadLongPtr
These are pointers to a single `int` or `long`. Compared to `NativeIntPtr` and `NativeLongPtr`, their `Parallel` versions are much faster to use in `ParallelFor` jobs but use more memory. Here's how to use `NativePerJobThreadIntPtr` (`NativePerJobThreadLongPtr` is identical):
```csharp
// Construct with the zero value
NativePerJobThreadIntPtr intPtr0 = new NativePerJobThreadIntPtr(Allocator.Temp);// Construct with a custom value
NativePerJobThreadIntPtr intPtr = new NativePerJobThreadIntPtr(Allocator.Temp, 123);// Read and write the value
intPtr.Value = 20;
Debug.Log("Value: " + intPtr.Value); // prints "Value: 20"// Get a Parallel for use in an IJobParallelFor
NativePerJobThreadIntPtr.Parallel parallel = intPtr.GetParallel();// Perform atomic writes on it
parallel.Increment();
parallel.Decrement();
parallel.Add(100);// Dispose the native memory
intPtr0.Dispose();
intPtr.Dispose();
```To read about the making of this type, see this [article](https://jacksondunstan.com/articles/4942).
# IJobParallelForRanged
This is a job type interface similar to `IJobParallelFor` for job types that want to operate on a range of indices from the batch rather than one index at a time. This is especially useful with `NativeChunkedList` as its enumerable and enumerator types can iterate much more efficiently than with random access via the indexer. Here's how to use this job type interface:
```csharp
[BurstCompile]
struct ChunkedListIterateJobParallelFor : IJobParallelForRanged
{
public NativeChunkedList List;
public NativePerJobThreadIntPtr.Parallel Sum;public void Execute(int startIndex, int endIndex)
{
for (
var chunks = List.GetChunksEnumerable(startIndex, endIndex).GetEnumerator();
chunks.MoveNext();)
{
for (
var chunk = chunks.Current.GetEnumerator();
chunk.MoveNext();)
{
Sum.Add(chunk.Current);
}
}
}
}
```To read about the making of this type, see this [article](https://jacksondunstan.com/articles/4976).
# SharedDisposable<T>
This allows for easy sharing of `IDisposable` objects such as the above collection types. A reference count is held internally and the object is free when its last reference is released. Here's how to use it:
```csharp
// Create an IDisposable
NativeArray array = new NativeArray(1, Allocator.Temp);// Prepare for sharing. Ref count = 1.
SharedDisposable> shared = array.Share(Allocator.Temp);// Share. Ref count = 2.
SharedDisposable> shared2 = shared.Ref();// Use a shared reference
print("Array len: " + shared2.Value.Length);// Release a reference. Ref count = 1.
shared2.Dispose();// Release the last reference. Ref count = 0.
// The NativeArray is disposed.
shared.Dispose();// 'using' blocks are even more convenient
// No need to call Dispose()
// Exception-safe
using (SharedDisposable> used = array.Share(Allocator.Temp))
{
using (SharedDisposable> used2 = shared.Ref())
{
}
}
```To read about the making of this type, see this [article](https://jacksondunstan.com/articles/5441).
# License
[MIT](LICENSE.md)