2018-11-21 18:02:49 +01:00
|
|
|
|
using System;
|
|
|
|
|
|
|
|
|
|
namespace BrotliLib.Collections{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// A ring buffer implemented as a fixed-size queue.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public sealed class RingBuffer<T>{
|
2020-04-07 17:37:52 +02:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Initializes a new ring buffer, with the values and size of the provided <paramref name="values"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="values">Initial values placed into the buffer. Element at [0] is at the back of the queue.</param>
|
|
|
|
|
public static RingBuffer<T> From(params T[] values){
|
|
|
|
|
return new RingBuffer<T>(values);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-21 18:02:49 +01:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Amount of elements in the buffer.
|
|
|
|
|
/// </summary>
|
2019-07-08 23:36:10 +02:00
|
|
|
|
public int Length { get; }
|
2018-11-21 18:02:49 +01:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Element at index [<see cref="Length"/> - 1].
|
|
|
|
|
/// </summary>
|
2019-07-08 23:36:10 +02:00
|
|
|
|
public T Front => this[Length - 1];
|
2018-11-21 18:02:49 +01:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Element at index [0].
|
|
|
|
|
/// </summary>
|
|
|
|
|
public T Back => this[0];
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Element at the provided <paramref name="index"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <exception cref="IndexOutOfRangeException">Thrown when the provided <paramref name="index"/> is negative.</exception>
|
|
|
|
|
public T this[int index]{
|
|
|
|
|
get{
|
|
|
|
|
if (index < 0){
|
|
|
|
|
throw new IndexOutOfRangeException("Ring buffer index cannot be negative.");
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-08 23:36:10 +02:00
|
|
|
|
return values[(index + accessOffset) % Length];
|
2018-11-21 18:02:49 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private readonly T[] values;
|
|
|
|
|
private int accessOffset;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Initializes a new ring buffer, with the values and size of the provided <paramref name="values"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="values">Initial values placed into the buffer. Element at [0] is at the back of the queue.</param>
|
2020-04-07 17:37:52 +02:00
|
|
|
|
public RingBuffer(T[] values){
|
2018-11-21 18:02:49 +01:00
|
|
|
|
if (values.Length == 0){
|
|
|
|
|
throw new ArgumentException("Ring buffer size cannot be zero.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.values = values;
|
2019-07-08 23:36:10 +02:00
|
|
|
|
this.Length = values.Length;
|
2018-11-21 18:02:49 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-03-28 03:31:28 +01:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Initializes a new ring buffer, with a shallow copy of the contents of the provided <paramref name="original"/>.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public RingBuffer(RingBuffer<T> original){
|
|
|
|
|
this.values = (T[])original.values.Clone();
|
2019-05-19 22:46:31 +02:00
|
|
|
|
this.accessOffset = original.accessOffset;
|
2019-07-08 23:36:10 +02:00
|
|
|
|
this.Length = original.Length;
|
2019-03-28 03:31:28 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-11-21 18:02:49 +01:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Pushes a new value to the front of the queue, removing the value at the back.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public void Push(T value){
|
2019-07-08 23:36:10 +02:00
|
|
|
|
values[accessOffset++ % Length] = value;
|
2018-11-21 18:02:49 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|