Advertisement
apieceoffruit

IndexedCollection

Mar 5th, 2025
314
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
C# 4.19 KB | None | 0 0
  1. public class IndexedCollection<T> : IEnumerable<T> where T : class
  2.     {
  3.         int _count;
  4.         int _freeCount;
  5.         int[] _freeIndices;
  6.         readonly bool _isRented;
  7.         T[] _items;
  8.  
  9.         public IndexedCollection(int capacity = 64)
  10.         {
  11.             Capacity = capacity;
  12.             if (capacity > 1024)
  13.             {
  14.                 _items = ArrayPool<T>.Shared.Rent(capacity);
  15.                 _freeIndices = ArrayPool<int>.Shared.Rent(capacity);
  16.                 _isRented = true;
  17.             }
  18.             else
  19.             {
  20.                 _items = new T[capacity];
  21.                 _freeIndices = new int[capacity];
  22.             }
  23.  
  24.             _count = 0;
  25.             _freeCount = 0;
  26.         }
  27.  
  28.         public int Count => _count - _freeCount;
  29.  
  30.         public int Capacity { get; private set; }
  31.  
  32.         public int Add(T item)
  33.         {
  34.             int index;
  35.  
  36.             if (_freeCount > 0)
  37.             {
  38.                 index = _freeIndices[--_freeCount];
  39.                 _items[index] = item;
  40.             }
  41.             else
  42.             {
  43.                 if (_count == Capacity) Expand();
  44.  
  45.                 index = _count++;
  46.                 _items[index] = item;
  47.             }
  48.  
  49.             return index;
  50.         }
  51.        
  52.         public T this[int index] => Get(index);
  53.  
  54.         public T Get(int index)
  55.         {
  56.             if (index < 0 || index >= _count) throw new ArgumentOutOfRangeException(nameof(index));
  57.             return _items[index];
  58.         }
  59.  
  60.         public bool TryGet(int index, out T item)
  61.         {
  62.             if (index >= 0 && index < _count)
  63.             {
  64.                 item = _items[index];
  65.                 return item != null;
  66.             }
  67.             item = null;
  68.             return false;
  69.         }
  70.  
  71.         public void Remove(int index, bool compact = false)
  72.         {
  73.             if (index < 0 || index >= _count) throw new ArgumentOutOfRangeException(nameof(index));
  74.  
  75.             _items[index] = null;
  76.  
  77.             if (compact && index == _count - 1)
  78.             {
  79.                 _count--;
  80.                 while (_count > 0 && _items[_count - 1] == null) _count--;
  81.                 _freeCount = Math.Max(0, _freeCount - (_count - index));
  82.             }
  83.             else
  84.                 _freeIndices[_freeCount++] = index;
  85.         }
  86.  
  87.         public void Clear()
  88.         {
  89.             for (var i = 0; i < _count; i++) _items[i] = null;
  90.             _count = 0;
  91.             _freeCount = 0;
  92.         }
  93.  
  94.         public IEnumerable<T> GetActiveItems()
  95.         {
  96.             for (var i = 0; i < _count; i++)
  97.                 if (_items[i] != null)
  98.                     yield return _items[i];
  99.         }
  100.  
  101.         void Expand()
  102.         {
  103.             var newCapacity = Capacity * 2;
  104.  
  105.             T[] newItems;
  106.             int[] newFreeIndices;
  107.  
  108.             if (_isRented && newCapacity > 1024)
  109.             {
  110.                 newItems = ArrayPool<T>.Shared.Rent(newCapacity);
  111.                 newFreeIndices = ArrayPool<int>.Shared.Rent(newCapacity);
  112.                 Array.Copy(_items, 0, newItems, 0, _count);
  113.                 Array.Copy(_freeIndices, 0, newFreeIndices, 0, _freeCount);
  114.                 ArrayPool<T>.Shared.Return(_items);
  115.                 ArrayPool<int>.Shared.Return(_freeIndices);
  116.             }
  117.             else
  118.             {
  119.                 newItems = new T[newCapacity];
  120.                 newFreeIndices = new int[newCapacity];
  121.                 Array.Copy(_items, 0, newItems, 0, _count);
  122.                 Array.Copy(_freeIndices, 0, newFreeIndices, 0, _freeCount);
  123.             }
  124.  
  125.             _items = newItems;
  126.             _freeIndices = newFreeIndices;
  127.             Capacity = newCapacity;
  128.         }
  129.  
  130.         public void Dispose()
  131.         {
  132.             if (_isRented)
  133.             {
  134.                 Array.Clear(_items, 0, _count);
  135.                 ArrayPool<T>.Shared.Return(_items);
  136.                 ArrayPool<int>.Shared.Return(_freeIndices);
  137.             }
  138.  
  139.             _items = null;
  140.             _freeIndices = null;
  141.         }
  142.  
  143.         public IEnumerator<T> GetEnumerator() => GetActiveItems().GetEnumerator();
  144.  
  145.         IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
  146.     }
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement