using System.Collections; namespace Persistence.Repository; /// /// Цикличный массив /// /// public class CyclicArray : IEnumerable { readonly T[] array; int used, current = -1; /// /// constructor /// /// public CyclicArray(int capacity) { array = new T[capacity]; } /// /// Количество элементов в массиве /// public int Count => used; /// /// Добавить новый элемент
/// Если capacity достигнуто, то вытеснит самый первый элемент ///
/// public void Add(T item) { current = (++current) % array.Length; array[current] = item; if (used < array.Length) used++; UpdatedInvoke(current, item); } /// /// Добавить новые элементы.
/// Если capacity достигнуто, то вытеснит самые первые элементы.
/// Не вызывает Updated! ///
/// public void AddRange(IEnumerable items) { var capacity = array.Length; var newItems = items.TakeLast(capacity).ToArray(); if (newItems.Length == capacity) { Array.Copy(newItems, array, capacity); current = capacity - 1; } else { current = (++current) % capacity; var countToEndOfArray = capacity - current; if (newItems.Length <= countToEndOfArray) { Array.Copy(newItems, 0, array, current, newItems.Length); current += newItems.Length - 1; } else { var firstStepLength = countToEndOfArray; Array.Copy(newItems, 0, array, current, firstStepLength); var secondStepCount = newItems.Length - firstStepLength; Array.Copy(newItems, firstStepLength, array, 0, secondStepCount); current = secondStepCount - 1; } } if (used < capacity) { used += newItems.Length; used = used > capacity ? capacity : used; } } /// /// Индекс /// /// /// public T this[int index] { get { if (used == 0) throw new IndexOutOfRangeException(); var i = (current + 1 + index) % used; return array[i]; } set { var devider = used > 0 ? used : array.Length; var i = (current + 1 + index) % devider; array[i] = value; UpdatedInvoke(current, value); } } /// /// событие на изменение элемента в массиве /// public event EventHandler<(int index, T value)>? Updated; private void UpdatedInvoke(int index, T value) { Updated?.Invoke(this, (index, value)); } /// /// Агрегирование значения по всему массиву /// /// /// /// /// public Tout Aggregate(Func func, Tout startValue) { Tout result = startValue; for (int i = 0; i < used; i++) result = func(this[i], result); return result; } /// public IEnumerator GetEnumerator() => new CyclicListEnumerator(array, current, used); /// IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); class CyclicListEnumerator : IEnumerator { private readonly Te[] array; private readonly int used; private readonly int first; private int current = -1; public CyclicListEnumerator(Te[] array, int first, int used) { this.array = new Te[array.Length]; array.CopyTo(this.array, 0); this.used = used; this.first = first; } public Te Current { get { if (IsCurrentOk()) { var i = (current + first + 1) % used; return array[i]; } else return default!; } } object? IEnumerator.Current => Current; public void Dispose() {; } private bool IsCurrentOk() => current >= 0 && current < used; public bool MoveNext() { if (current < used) current++; return IsCurrentOk(); } public void Reset() { current = -1; } } /// /// Очистить весь массив /// public void Clear() { used = 0; current = -1; } }