深入理解 C#.NET IEnumerable<T>:一切集合的起点

简介

IEnumerable<T>.NET 中最核心的接口之一,位于 System.Collections.Generic 命名空间中。它代表一个可枚举的集合,支持在集合上进行迭代操作。

IEnumerable<T> 是什么?

csharp 复制代码
public interface IEnumerable<out T> : IEnumerable
{
    IEnumerator<T> GetEnumerator();
}
  • 它定义了一个可枚举对象的契约;

  • 任何实现了 IEnumerable<T> 的类型都能被 foreach 循环遍历;

  • 泛型版 IEnumerable<T> 是非泛型 IEnumerable 的类型安全扩展;

  • 它的核心方法只有一个:GetEnumerator()

IEnumerable 与 IEnumerator 的关系

要理解 IEnumerable<T>,必须知道它依赖另一个接口:

csharp 复制代码
public interface IEnumerator<out T> : IDisposable, IEnumerator
{
    T Current { get; }
    bool MoveNext();
    void Reset(); // 通常不实现
}

关系示意图:

scss 复制代码
IEnumerable<T>
    └── GetEnumerator()
          └── 返回 IEnumerator<T>
                  ├── MoveNext() → 是否还有元素
                  ├── Current → 当前元素
                  └── Reset() → 重置(可选)

基本用法

csharp 复制代码
using System;
using System.Collections.Generic;

class Program
{
    static void Main()
    {
        List<string> fruits = new List<string> { "Apple", "Banana", "Cherry" };
        
        // 使用 foreach 遍历(推荐)
        foreach (string fruit in fruits)
        {
            Console.WriteLine(fruit);
        }
        
        // 等价的手动迭代方式
        IEnumerator<string> enumerator = fruits.GetEnumerator();
        try
        {
            while (enumerator.MoveNext())
            {
                string fruit = enumerator.Current;
                Console.WriteLine(fruit);
            }
        }
        finally
        {
            enumerator?.Dispose();
        }
    }
}

手写一个 IEnumerable<T> 示例

csharp 复制代码
public class NumberCollection : IEnumerable<int>
{
    private readonly int[] _numbers = { 1, 2, 3, 4, 5 };

    public IEnumerator<int> GetEnumerator()
    {
        foreach (var n in _numbers)
            yield return n;
    }

    IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
}

使用:

csharp 复制代码
var numbers = new NumberCollection();
foreach (var n in numbers)
{
    Console.WriteLine(n);
}

输出:

复制代码
1
2
3
4
5

yield return 的魔法

等价于下面的展开版:

csharp 复制代码
public IEnumerator<int> GetEnumerator()
{
    return new Enumerator();
}

private class Enumerator : IEnumerator<int>
{
    private int _index = -1;
    private readonly int[] _numbers = { 1, 2, 3, 4, 5 };

    public int Current => _numbers[_index];
    object IEnumerator.Current => Current;

    public bool MoveNext()
    {
        _index++;
        return _index < _numbers.Length;
    }

    public void Reset() => _index = -1;
    public void Dispose() { }
}

yield 编译后自动生成状态机类,就是这种效果。

这也是 LINQ 延迟执行的基础机制。

foreach 的底层原理

当写:

csharp 复制代码
foreach (var item in collection)
{
    Console.WriteLine(item);
}

编译器实际上生成:

csharp 复制代码
using (var enumerator = collection.GetEnumerator())
{
    while (enumerator.MoveNext())
    {
        var item = enumerator.Current;
        Console.WriteLine(item);
    }
}

延迟执行(Lazy Evaluation)

LINQ 查询(Where, Select 等)通常返回 IEnumerable<T>。 这些操作是延迟执行的:只有在遍历时才真正运行。

csharp 复制代码
var numbers = new[] { 1, 2, 3, 4, 5 };

var query = numbers.Where(n => n > 2).Select(n => n * 10);

Console.WriteLine("Query created.");
foreach (var n in query)
{
    Console.WriteLine(n);
}

Where / Select 只是定义查询,不会立即执行。 直到 foreach 时,才会真正迭代并执行逻辑。

常见实现 IEnumerable<T> 的类型

类型 说明
List<T> 基于数组实现的集合
T[] 数组
Dictionary<TKey,TValue> 键值对集合(枚举键值对)
HashSet<T> 不重复元素集合
Queue<T> / Stack<T> 队列与栈
string 实现了非泛型 IEnumerable<char>
LINQ 查询结果 延迟执行序列

手写一个支持过滤的 IEnumerable<T>

csharp 复制代码
public class FilteredCollection<T> : IEnumerable<T>
{
    private readonly IEnumerable<T> _source;
    private readonly Func<T, bool> _predicate;

    public FilteredCollection(IEnumerable<T> source, Func<T, bool> predicate)
    {
        _source = source;
        _predicate = predicate;
    }

    public IEnumerator<T> GetEnumerator()
    {
        foreach (var item in _source)
        {
            if (_predicate(item))
                yield return item;
        }
    }

    IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
}

使用:

csharp 复制代码
var list = new List<int> { 1, 2, 3, 4, 5 };
var filtered = new FilteredCollection<int>(list, x => x % 2 == 0);

foreach (var n in filtered)
    Console.WriteLine(n);

IEnumerable<T> vs IQueryable<T>

特性 IEnumerable IQueryable
执行时机 本地内存中 可翻译为远程查询(如 SQL)
适用场景 内存集合 数据库 ORM(EF Core 等)
表达式类型 委托(Func) 表达式树(Expression)
可延迟执行
例子 List<T>, Array, yield return DbSet<T>

示例:

csharp 复制代码
// IEnumerable:在内存中过滤
var result1 = list.Where(x => x > 10);

// IQueryable:生成 SQL 查询
var result2 = db.Users.Where(x => x.Age > 10);

IEnumerable 的扩展方法分类(LINQ 常用)

分类 示例方法
过滤 Where, Distinct, Skip, Take
投影 Select, SelectMany
聚合 Count, Sum, Average, Aggregate
元素 First, Last, Single, ElementAt
组合 Concat, Union, Intersect, Except
排序 OrderBy, ThenBy, Reverse
转换 ToList, ToArray, ToDictionary

高级特性

自定义 LINQ 扩展方法

csharp 复制代码
public static class MyLinqExtensions
{
    // 自定义 Where 方法
    public static IEnumerable<T> Where<T>(
        this IEnumerable<T> source, 
        Func<T, bool> predicate)
    {
        foreach (T item in source)
        {
            if (predicate(item))
            {
                yield return item;
            }
        }
    }
    
    // 自定义 Select 方法
    public static IEnumerable<TResult> Select<TSource, TResult>(
        this IEnumerable<TSource> source,
        Func<TSource, TResult> selector)
    {
        foreach (TSource item in source)
        {
            yield return selector(item);
        }
    }
    
    // 自定义扩展方法
    public static IEnumerable<T> SkipEveryOther<T>(this IEnumerable<T> source)
    {
        bool take = true;
        foreach (T item in source)
        {
            if (take)
            {
                yield return item;
            }
            take = !take;
        }
    }
    
    // 带索引的扩展方法
    public static IEnumerable<TResult> SelectWithIndex<TSource, TResult>(
        this IEnumerable<TSource> source,
        Func<TSource, int, TResult> selector)
    {
        int index = 0;
        foreach (TSource item in source)
        {
            yield return selector(item, index);
            index++;
        }
    }
}

// 使用自定义扩展方法
var numbers = new List<int> { 1, 2, 3, 4, 5, 6, 7, 8 };
var result = numbers.SkipEveryOther(); // 返回 1, 3, 5, 7

var indexed = numbers.SelectWithIndex((num, idx) => $"Index {idx}: {num}");

无限序列

csharp 复制代码
public static class InfiniteSequences
{
    // 无限数字序列
    public static IEnumerable<int> InfiniteNumbers()
    {
        int i = 0;
        while (true)
        {
            yield return i++;
        }
    }
    
    // 斐波那契数列
    public static IEnumerable<long> Fibonacci()
    {
        long a = 0, b = 1;
        while (true)
        {
            yield return a;
            long temp = a;
            a = b;
            b = temp + b;
        }
    }
    
    // 随机数序列
    public static IEnumerable<int> RandomNumbers(int min, int max)
    {
        Random rnd = new Random();
        while (true)
        {
            yield return rnd.Next(min, max);
        }
    }
}

// 使用无限序列(一定要结合 Take 等方法使用)
var firstTenFibonacci = Fibonacci().Take(10);
var randomNumbers = RandomNumbers(1, 100).Take(5);

数据分页

csharp 复制代码
public static class PagingExtensions
{
    public static IEnumerable<IEnumerable<T>> Page<T>(this IEnumerable<T> source, int pageSize)
    {
        var page = new List<T>(pageSize);
        foreach (T item in source)
        {
            page.Add(item);
            if (page.Count == pageSize)
            {
                yield return page;
                page = new List<T>(pageSize);
            }
        }
        
        if (page.Count > 0)
        {
            yield return page;
        }
    }
}

// 使用分页
var bigCollection = Enumerable.Range(1, 1000);
foreach (var page in bigCollection.Page(100))
{
    Console.WriteLine($"Page with {page.Count()} items");
    // 处理当前页
}
相关推荐
richard_yuu12 分钟前
C#开发全景概述:从零读懂C#的定位、优势与完整技术体系
开发语言·c#
Xin_ye1008614 分钟前
C# 零基础到精通教程 - 第十二章:异常处理与调试——让程序更健壮
开发语言·c#
楼田莉子16 分钟前
C#学习之C#入门学习
开发语言·后端·学习·c#
小钻风336631 分钟前
从零入门 Kafka:Java 原生 API 到 Spring Boot 实战全解析
c#·linq
步步为营DotNet35 分钟前
解锁.NET 11 新境:ASP.NET Core 10 在微服务安全通信的深化与实践
微服务·asp.net·.net
唐青枫40 分钟前
C#.NET YARP + OpenTelemetry:网关链路追踪实战
c#·.net
Xin_ye1008613 小时前
C# 零基础到精通教程 - 第七章:面向对象编程(入门)——类与对象
开发语言·c#
rockey62713 小时前
AScript异步执行与await关键字
c#·.net·script·eval·expression·异步执行·动态脚本
叫我少年14 小时前
ASP.NET Core 最小 API 快速参考
.net·api
程序leo源15 小时前
Qt窗口详解
开发语言·数据库·c++·qt·青少年编程·c#