mirror of
https://github.com/chylex/.NET-Community-Toolkit.git
synced 2024-11-24 07:42:45 +01:00
112 lines
4.2 KiB
C#
112 lines
4.2 KiB
C#
// Licensed to the .NET Foundation under one or more agreements.
|
|
// The .NET Foundation licenses this file to you under the MIT license.
|
|
// See the LICENSE file in the project root for more information.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Runtime.CompilerServices;
|
|
|
|
namespace CommunityToolkit.HighPerformance.Enumerables;
|
|
|
|
/// <summary>
|
|
/// A <see langword="ref"/> <see langword="struct"/> that tokenizes a given <see cref="ReadOnlySpan{T}"/> instance.
|
|
/// </summary>
|
|
/// <typeparam name="T">The type of items to enumerate.</typeparam>
|
|
[EditorBrowsable(EditorBrowsableState.Never)]
|
|
public ref struct ReadOnlySpanTokenizer<T>
|
|
where T : IEquatable<T>
|
|
{
|
|
/// <summary>
|
|
/// The source <see cref="ReadOnlySpan{T}"/> instance.
|
|
/// </summary>
|
|
private readonly ReadOnlySpan<T> span;
|
|
|
|
/// <summary>
|
|
/// The separator item to use.
|
|
/// </summary>
|
|
private readonly T separator;
|
|
|
|
/// <summary>
|
|
/// The current initial offset.
|
|
/// </summary>
|
|
private int start;
|
|
|
|
/// <summary>
|
|
/// The current final offset.
|
|
/// </summary>
|
|
private int end;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="ReadOnlySpanTokenizer{T}"/> struct.
|
|
/// </summary>
|
|
/// <param name="span">The source <see cref="ReadOnlySpan{T}"/> instance.</param>
|
|
/// <param name="separator">The separator item to use.</param>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public ReadOnlySpanTokenizer(ReadOnlySpan<T> span, T separator)
|
|
{
|
|
this.span = span;
|
|
this.separator = separator;
|
|
this.start = 0;
|
|
this.end = -1;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Implements the duck-typed <see cref="IEnumerable{T}.GetEnumerator"/> method.
|
|
/// </summary>
|
|
/// <returns>An <see cref="ReadOnlySpanTokenizer{T}"/> instance targeting the current <see cref="ReadOnlySpan{T}"/> value.</returns>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public readonly ReadOnlySpanTokenizer<T> GetEnumerator() => this;
|
|
|
|
/// <summary>
|
|
/// Implements the duck-typed <see cref="System.Collections.IEnumerator.MoveNext"/> method.
|
|
/// </summary>
|
|
/// <returns><see langword="true"/> whether a new element is available, <see langword="false"/> otherwise</returns>
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
public bool MoveNext()
|
|
{
|
|
int newEnd = this.end + 1;
|
|
int length = this.span.Length;
|
|
|
|
// Additional check if the separator is not the last character
|
|
if (newEnd <= length)
|
|
{
|
|
this.start = newEnd;
|
|
|
|
// We need to call this extension explicitly or the extension method resolution rules for the C# compiler
|
|
// will end up picking CommunityToolkit.HighPerformance.ReadOnlySpanExtensions.IndexOf instead, even
|
|
// though the latter takes the parameter via a readonly reference. This is because the "in" modifier is
|
|
// implicit, which makes the signature compatible, and because extension methods are matched in such a
|
|
// way that methods "closest" to where they're used are preferred. Since this type shares the same root
|
|
// namespace, this makes that extension a better match, so that it overrides the MemoryExtensions one.
|
|
// This is not a problem for consumers of this package, as their code would be outside of the
|
|
// CommunityToolkit.HighPerformance namespace, so both extensions would be "equally distant", so that
|
|
// when they're both in scope it will be possible to choose which one to use by adding an explicit "in".
|
|
int index = System.MemoryExtensions.IndexOf(this.span.Slice(newEnd), this.separator);
|
|
|
|
// Extract the current subsequence
|
|
if (index >= 0)
|
|
{
|
|
this.end = newEnd + index;
|
|
|
|
return true;
|
|
}
|
|
|
|
this.end = length;
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the duck-typed <see cref="IEnumerator{T}.Current"/> property.
|
|
/// </summary>
|
|
public readonly ReadOnlySpan<T> Current
|
|
{
|
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
get => this.span.Slice(this.start, this.end - this.start);
|
|
}
|
|
}
|