yield return用于按需返回数据,延迟执行以提升性能;方法需返回IEnumerable或IEnumerator,每次迭代生成一个元素,避免一次性加载大量数据到内存。

yield return 是 C# 中一个非常实用但初学者容易忽略的关键字,它主要用于简化集合的迭代过程,延迟执行并按需返回数据。通过 yield return,开发者可以避免一次性将大量数据加载到内存中,从而提升性能和资源利用率。
yield return 基本语法与原理
使用 yield return 的方法必须返回类型为 IEnumerable
当调用方开始遍历这个方法返回的结果时(如使用 foreach),代码会从头执行到第一个 yield return,返回值后暂停,直到下一次迭代才继续执行。
示例:public IEnumerableGetNumbers()
{
for (int i = 1; i <= 5; i++)
{
yield return i;
}
}
调用时:
foreach (var num in GetNumbers())
{
Console.WriteLine(num);
}
输出结果是逐个打印 1 到 5,每次调用 yield return 返回一个值,而不是构建一个数组再返回。
yield return 的典型使用场景
在以下几种常见场景中,yield return 能显著提升效率和可读性:
1. 处理大型数据集或流式数据
当你需要读取大文件、数据库记录或网络流数据时,不希望一次性加载所有内容到内存。使用 yield return 可实现“边读边处理”。
例如:逐行读取文本文件
public IEnumerableReadLines(string filePath)
{
using var reader = new StreamReader(filePath);
string line;
while ((line = reader.ReadLine()) != null)
{
yield return line;
}
}
这样每行数据只有在被枚举时才会读取,节省内存。
2. 实现自定义迭代逻辑
比如生成斐波那契数列、树结构遍历、无限序列等,可以用简洁代码表达复杂迭代过程。
示例:生成斐波那契数列前 n 项
public IEnumerableFibonacci(int count)
{
int current = 0, next = 1;
for (int i = 0; i < count; i++)
{
yield return current;
int temp = current;
current = next;
next += temp;
}
}
3. 条件过滤与懒加载集合
结合条件判断,只在满足条件时返回元素,适用于筛选操作。
例如:返回偶数
public IEnumerableGetEvens(IEnumerable source)
{
foreach (var item in source)
{
if (item % 2 == 0)
yield return item;
}
}
这种写法比先创建 List 再 Add 更高效,尤其是源数据很大时。
注意事项与限制
虽然 yield return 很方便,但也有一些限制需要注意:
- 不能用于异步方法(async)中,即不能在 async 方法里使用 yield return
- 异常处理需谨慎,因为实际执行时机可能延迟到枚举时才触发
- 无法直接获取总数 Count,除非完全枚举一遍
- 多次枚举会重新执行逻辑,不适合有副作用的操作
如果需要支持异步流,.NET 6+ 提供了 IAsyncEnumerable
总结
yield return 是一种实现惰性求值的强大工具,适合用于集合生成、大数据处理和自定义迭代器。它让代码更简洁、内存更友好,特别适合“边生成边消费”的场景。掌握其使用方式和边界条件,能有效提升 .NET 应用的性能与可维护性。
基本上就这些,关键在于理解“按需返回”这一核心思想。










