ListSegment<T>
public struct ListSegment<T> : IList<T>, ICollection<T>, IEnumerable<T>, IEnumerable, IReadOnlyList<T>, IReadOnlyCollection<T>
Delimits a section of a list.
using System;
using System.Collections;
using System.Collections.Generic;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
namespace DotNext.Collections.Generic
{
[StructLayout(LayoutKind.Auto)]
[NullableContext(1)]
[Nullable(0)]
public readonly struct ListSegment<[Nullable(2)] T> : IList<T>, ICollection<T>, IEnumerable<T>, IEnumerable, IReadOnlyList<T>, IReadOnlyCollection<T>
{
private readonly int startIndex;
private readonly IList<T> list;
bool ICollection<T>.IsReadOnly {
get {
return true;
}
}
public int Count { get; }
public T this[int index] {
get {
return list[ToAbsoluteIndex(index)];
}
set {
list[ToAbsoluteIndex(index)] = value;
}
}
public ListSegment(IList<T> list, Range range)
{
(int, int) offsetAndLength = range.GetOffsetAndLength(list.Count);
startIndex = offsetAndLength.Item1;
Count = offsetAndLength.Item2;
this.list = list;
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private int ToAbsoluteIndex(int index)
{
if (!BasicExtensions.IsBetween<int, EnclosedEndpoint<int>, DisclosedEndpoint<int>>(index, BasicExtensions.Enclosed<int>(0), BasicExtensions.Disclosed<int>(Count)))
throw new ArgumentOutOfRangeException("index");
return index + startIndex;
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private bool ToRelativeIndex(ref int index)
{
index -= startIndex;
return BasicExtensions.IsBetween<int, EnclosedEndpoint<int>, DisclosedEndpoint<int>>(index, BasicExtensions.Enclosed<int>(0), BasicExtensions.Disclosed<int>(Count));
}
public int IndexOf(T item)
{
int index = list.IndexOf(item);
if (!ToRelativeIndex(ref index))
return -1;
return index;
}
public bool TryGetSpan([Nullable(new byte[] {
0,
1
})] out Span<T> span)
{
IList<T> list = this.list;
List<T> list2 = list as List<T>;
if (list2 == null) {
T[] array = list as T[];
if (array == null) {
span = default(Span<T>);
return false;
}
span = new Span<T>(array, startIndex, Count);
} else
span = CollectionsMarshal.AsSpan<T>(list2).Slice(startIndex, Count);
return true;
}
void IList<T>.Insert(int index, T item)
{
throw new NotSupportedException();
}
void IList<T>.RemoveAt(int index)
{
throw new NotSupportedException();
}
void ICollection<T>.Add(T item)
{
throw new NotSupportedException();
}
void ICollection<T>.Clear()
{
throw new NotSupportedException();
}
public bool Contains(T item)
{
return IndexOf(item) >= 0;
}
public void CopyTo(T[] array, int arrayIndex)
{
IList<T> list = this.list;
List<T> list2 = list as List<T>;
if (list2 == null) {
T[] array2 = list as T[];
if (array2 != null)
Array.Copy(array2, startIndex, array, arrayIndex, Count);
else {
for (int i = 0; i < Count; i++) {
array[i] = this.list[i + startIndex];
}
}
} else
list2.CopyTo(startIndex, array, arrayIndex, Count);
}
bool ICollection<T>.Remove(T item)
{
throw new NotSupportedException();
}
public IEnumerator<T> GetEnumerator()
{
IEnumerator<T> enumerator = list.GetEnumerator();
Enumerator.Skip<T>(enumerator, startIndex);
return Enumerator.Limit<T>(enumerator, Count, false);
}
IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
}
}