0

0

C# Yield关键字方法 C#如何使用yield return实现迭代器

幻夢星雲

幻夢星雲

发布时间:2026-01-22 08:45:22

|

748人浏览过

|

来源于php中文网

原创

yield return 是 C# 中用于定义迭代器方法的关键字,它不终止方法而是暂停执行并返回一个值,由编译器生成状态机实现 IEnumerable;必须返回 IEnumerable 等类型,不可混用普通 return,且受 try/catch 和闭包陷阱等限制。

c# yield关键字方法 c#如何使用yield return实现迭代器

yield return 是什么,它不是返回值而是挂起点

yield return 不是普通函数的 return,它不会终止方法执行,而是暂停当前迭代器状态,把值“交出去”,等下一次调用 MoveNext() 时从暂停处继续。这意味着方法体实际被编译成一个实现了 IEnumerableIEnumerator 的状态机类,你写的代码只是语法糖。

常见误解是把它当“逐个 return”,结果在循环里写了 return 混用,导致后续 yield return 永远不执行 —— 这种写法直接报错或逻辑中断。

  • 方法返回类型必须是 IEnumerableIEnumerableIAsyncEnumerable(C# 8+)之一
  • 方法体内不能有普通 return 语句(除了 return; 用于提前退出迭代)
  • 不能在 try 块中有 yield return,除非 catchfinally 中没有 yield return(编译器限制)

怎么写一个基础 yield return 迭代器方法

最典型场景:把一个计算过程或数据流封装成可枚举对象,避免一次性加载全部数据到内存。

public static IEnumerable GetEvenNumbers(int max)
{
    for (int i = 0; i <= max; i += 2)
    {
        yield return i;
    }
}

调用时:

foreach (int n in GetEvenNumbers(10))
{
    Console.WriteLine(n); // 输出 0, 2, 4, 6, 8, 10
}

注意:GetEvenNumbers(10) 调用本身不执行循环,只返回一个未启动的迭代器;真正执行从 foreach 第一次调用 MoveNext() 开始。

  • 每次 yield return 后,方法暂停,局部变量(如 i)状态被保留
  • 如果想中途退出,用 yield break;,它相当于“迭代结束”,不是异常
  • 不要在 yield return 后写任何代码(除非是 yield break; 或空语句),编译器会报错

yield return 和 List.Add 的性能与内存差异

对比两种实现方式:

MCP Market
MCP Market

MCP Servers集合平台,帮你找到最好的MCP服务器

下载
// ❌ 先构造完整列表再返回
public static List GetEvenNumbersList(int max)
{
    var list = new List();
    for (int i = 0; i <= max; i += 2)
    {
        list.Add(i);
    }
    return list;
}
// ✅ yield return 流式生成
public static IEnumerable GetEvenNumbers(int max)
{
    for (int i = 0; i <= max; i += 2)
    {
        yield return i;
    }
}

关键区别

  • List 版本必须分配足够内存容纳所有元素(比如 max = 1000000 就要存 50 万整数),且全部算完才返回
  • yield return 版本按需计算,内存占用恒定(仅保存当前状态),适合大数据流、IO 边界(如逐行读文件)、或消费者可能提前退出的场景(如 .FirstOrDefault()
  • 但无法重复遍历(除非显式调用 .ToList()),因为每次调用都新建迭代器

容易踩的坑:闭包捕获和延迟执行陷阱

下面这段代码很常见,但结果不符合直觉:

public static IEnumerable> GetDelegates()
{
    var actions = new List>();
    for (int i = 0; i < 3; i++)
    {
        yield return () => i; // ❌ 所有委托都返回 3
    }
}

原因:所有 yield return 返回的 lambda 共享同一个变量 i,而迭代器直到遍历时才执行,此时循环早已结束,i == 3

修复方式:在循环内创建局部副本:

for (int i = 0; i < 3; i++)
{
    int localI = i; // ✅ 每次迭代独立副本
    yield return () => localI;
}

另一个坑是误以为 yield return 方法“立即执行”——它其实完全惰性。如果你在 yield return 方法里打开文件、数据库连接或 HTTP 请求,这些资源会在第一次 MoveNext() 时才初始化,且若没正确释放(比如没用 using 包裹),容易造成资源泄漏或并发问题。

相关专题

更多
php中foreach用法
php中foreach用法

本专题整合了php中foreach用法的相关介绍,阅读专题下面的文章了解更多详细教程。

47

2025.12.04

java中break的作用
java中break的作用

本专题整合了java中break的用法教程,阅读专题下面的文章了解更多详细内容。

118

2025.10.15

java break和continue
java break和continue

本专题整合了java break和continue的区别相关内容,阅读专题下面的文章了解更多详细内容。

256

2025.10.24

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

205

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

50

2026.01.05

go语言闭包相关教程大全
go语言闭包相关教程大全

本专题整合了go语言闭包相关数据,阅读专题下面的文章了解更多相关内容。

136

2025.07.29

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

352

2023.06.29

Golang 性能分析与pprof调优实战
Golang 性能分析与pprof调优实战

本专题系统讲解 Golang 应用的性能分析与调优方法,重点覆盖 pprof 的使用方式,包括 CPU、内存、阻塞与 goroutine 分析,火焰图解读,常见性能瓶颈定位思路,以及在真实项目中进行针对性优化的实践技巧。通过案例讲解,帮助开发者掌握 用数据驱动的方式持续提升 Go 程序性能与稳定性。

6

2026.01.22

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
CSS3 教程
CSS3 教程

共18课时 | 4.8万人学习

PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.6万人学习

Django 教程
Django 教程

共28课时 | 3.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号