Span<T>.Enumerator 结构

定义

为 . 的 Span<T>元素提供枚举器。

public: value class Span<T>::Enumerator : System::Collections::Generic::IEnumerator<T>
public: value class Span<T>::Enumerator
public ref struct Span<T>.Enumerator : System.Collections.Generic.IEnumerator<T>
public ref struct Span<T>.Enumerator
type Span<'T>.Enumerator = struct
    interface IEnumerator<'T>
    interface IEnumerator
    interface IDisposable
type Span<'T>.Enumerator = struct
Public Structure Span(Of T).Enumerator
Implements IEnumerator(Of T)
Public Structure Span(Of T).Enumerator

类型参数

T
继承
Span<T>.Enumerator
实现

注解

C# foreach c# 语言和 For Each...接下来构造Visual Basic隐藏枚举器的复杂性。 建议使用或For Each...Next建议直接foreach操作枚举器, 而不是直接操作枚举器。

最初,枚举器位于第一个元素之前 Span<T>。 在此位置, Current 未定义。 在读取值Current之前,必须调用MoveNext枚举器将枚举器前进到第一项Span<T>

Current 返回相同的值,直到 MoveNext 调用为止。 MoveNext 设置为 Current .. 中的 Span<T>下一项。

如果 MoveNext 传递末尾 Span<T>MoveNextfalse返回。 当枚举器处于此状态时,后续调用 MoveNext 也会返回 falseCurrent 未定义。 不能再次设置为 Current 第一项 Span<T> ;必须改为创建新的枚举器实例。

枚举器没有对 . Span<T>. 此外,还可以修改跨度所基于的基础数据。 因此,通过范围枚举本质上不是线程安全的过程。 若要保证枚举期间的线程安全性,必须实现自己的同步。 例如,以下代码具有争用条件。 它不确保在方法执行之前 ClearContents 枚举范围。 因此,在枚举范围期间清除基础数组:

using System;
using System.Threading.Tasks;

class Program
{
    private static readonly byte[] _array = new byte[5];

    static void Main()
    {
        new Random(42).NextBytes(_array);
        Span<byte> span = _array;

        Task.Run( () => ClearContents() );

       EnumerateSpan(span);
    }

    public static void ClearContents()
    {
        Task.Delay(20).Wait();
        lock (_array)
        {
           Array.Clear(_array, 0, _array.Length);
        }
    }

    public static void EnumerateSpan(Span<byte> span)
    {
        foreach (byte element in span)
        {
            Console.WriteLine(element);
            Task.Delay(10).Wait();
        }
    }
}
// The example displays output like the following:
//     62
//     23
//     186
//     0
//     0
module Program

open System
open System.Threading.Tasks

let array = Array.zeroCreate<byte> 5

let clearContents () =
    Task.Delay(20).Wait()
    lock array (fun () -> 
        Array.Clear(array, 0, array.Length) )

let enumerateSpan (span: Span<byte>) =
    for element in span do
        printfn $"{element}"
        Task.Delay(10).Wait()

[<EntryPoint>]
let main _ =
    Random(42).NextBytes array
    printfn "%A" array
    let span: Span<byte> = array

    Task.Run clearContents |> ignore

    enumerateSpan span
    
    0

// The example displays output like the following:
//     62
//     23
//     186
//     0
//     0

如果在枚举范围之前同步对数组的访问,如以下示例中方法的修改版本 EnumerateSpan 一样,该方法 ClearContents 不会在枚举期间修改基础范围数据。 请注意,该示例锁定范围所基于的基础数组。

public static void EnumerateSpan(Span<byte> span)
{
    lock (_array)
    {
        foreach (byte element in span)
        {
            Console.WriteLine(element);
            Task.Delay(10).Wait();
        }
    }
}
// The example displays the following output:
//    62
//    23
//    186
//    150
//    174
let enumerateSpan (span: Span<byte>) =
    // Spans cannot be accessed in closures including in the F# lock function.
    // Monitor.Enter and Monitor.Exit are used here directly.
    Monitor.Enter array
    try
        for element in span do
            printfn $"{element}"
            Task.Delay(10).Wait()
    finally
        Monitor.Exit array
// The example displays the following output:
//    62
//    23
//    186
//    150
//    174

与.NET中的一些其他枚举器结构不同,Span<T>.Enumerator

属性

名称 说明
Current

获取对枚举器当前位置的项的引用。

方法

名称 说明
MoveNext()

将枚举器前进到下 Span<T>一项。

显式接口实现

名称 说明
IDisposable.Dispose()

执行与释放、释放或重置非托管资源关联的应用程序定义任务。

IEnumerator.Current

获取集合中枚举器当前位置的元素。

IEnumerator.Reset()

将枚举器设置为其初始位置,该位置位于集合中的第一个元素之前。

IEnumerator<T>.Current

获取集合中枚举器当前位置的元素。

适用于