阅读本文大概需要 3 分钟。

大家好,这是 [C#.NET 拾遗补漏] 系列的第 07 篇文章。

在 C# 中,大多数方法都是通过 return 语句立即把程序的控制权交回给调用者,同时也会把方法内的本地资源释放掉。而包含 yield 语句的方法则允许在依次返回多个值给调用者的期间保留本地资源,等所有值都返回结束时再释放掉本来资源,这些返回的值形成一组序列被调用者使用。在 C# 中,这种包含 yield 语句的方法、属性或索引器就是迭代器。

迭代器中的 yield 语句分为两种:

  • yeild return,把程序控制权交回调用者并保留本地状态,调用者拿到返回的值继续往后执行。

  • yeild break,用于告诉程序当前序列已经结束,相当于正常代码块的 return 语句(迭代器中直接使用 return 是非法的)。

下面是一个用来生成斐波纳契序列的迭代器示例:

IEnumerable<int> Fibonacci(int count)
{int prev = 1;int curr = 1;for (int i = 0; i < count; i++){yield return prev;int temp = prev + curr;prev = curr;curr = temp;}
}void Main()
{foreach (int term in Fibonacci(10)){Console.WriteLine(term);}
}

输出:

1
1
2
3
5
8
13
21
34
55

实际场景中,我们一般很少直接写迭代器,因为大部分需要迭代的场景都是数组、集合和列表,而这些类型内部已经封装好了所需的迭代器。比如 C# 中的数组之所以可以被遍历是因为它实现了 IEnumerable 接口,通过 GetEnumerator() 方法可以获得数组的列举器 Enumerator,而该列举器就是通过迭代器来实现的。比如最常见的一种使用场景就是遍历数组中的每一个元素,如下面逐个打印数组元素的示例。

int[] numbers = { 1, 2, 3, 4, 5 };
IEnumerator enumerator = numbers.GetEnumerator();
while (enumerator.MoveNext())
{Console.WriteLine(enumerator.Current);
}

其实这就是 foreach 的工作原理,上面代码可以用 foreach 改写如下:

int[] numbers = { 1, 2, 3, 4, 5 };
foreach (int number in numbers)
{Console.WriteLine(number);
}

当然,列举器不一定非要通过迭代器实现,例如下面这个自定义的列举器 CoffeeEnumerator。

public class CoffeeCollection : IEnumerable
{private CoffeeEnumerator enumerator;public CoffeeCollection(){enumerator = new CoffeeEnumerator();}public IEnumerator GetEnumerator(){return enumerator;}public class CoffeeEnumerator : IEnumerator{string[] items = new string[3] { "espresso", "macchiato", "latte" };int currentIndex = -1;public object Current{get{return items[currentIndex];}}public bool MoveNext(){currentIndex++;if (currentIndex < items.Length){return true;}return false;}public void Reset(){currentIndex = 0;}}
}

使用:

public static void Main(string[] args)
{foreach (var coffee in new CoffeeCollection()){Console.WriteLine(coffee);}
}

理解迭代器和列举器可以帮助我们写出更高效的代码。比如判断一个 IEnumerable<T> 对象是否包含元素,经常看到有些人这么写:

if(enumerable.Count() > 0)
{// 集合中有元素
}

但如果用列举器的思维稍微思考一下就知道,Count() 为了获得集合元素数量必然要迭代完所有元素,时间复杂度为 O(n)。而仅仅是要知道集合中是否包含元素,其实迭代一次就可以了。所以效率更好的做法是:

if(enumerable.GetEnumerator().MoveNext())
{// 集合中有元素
}

这样写时间复杂度是 O(1),效率显然更高。为了书写方便,C# 提供了扩展方法 Any()

if(enumerable.Any())
{// 集合中有元素
}

所以如有需要,应尽可能使用 Any 方法,效率更高。

再比如在 EF Core 中,需要执行 IQueryable<T> 查询时,有时候使用 AsEnumerable() 比使用 ToList、ToArray 等更高效,因为 ToList、ToArray 等会立即执行列举操作,而 AsEnumerable() 可以把列举操作延迟到真正被需要的时候再执行。当然也要考虑实际应用场景,Array、List 等更方便调用者使用,特别是要获取元素总数量、增删元素等这种操作。

[C#.NET 拾遗补漏]07:迭代器和列举器相关推荐

  1. c# xml文件新增同级节点_[C#.NET 拾遗补漏]08:强大的LINQ

    大家好,这是 [C#.NET 拾遗补漏] 系列的第 08 篇文章,今天讲 C# 强大的 LINQ 查询.LINQ 是我最喜欢的 C# 语言特性之一. LINQ 是 Language INtegrate ...

  2. [C#.NET 拾遗补漏]08:强大的LINQ

    阅读本文大概需要 13 分钟. 大家好,这是 [C#.NET 拾遗补漏] 系列的第 08 篇文章,今天讲 C# 强大的 LINQ 查询.LINQ 是我最喜欢的 C# 语言特性之一. LINQ 是 La ...

  3. [C#.NET 拾遗补漏]06:单例模式最佳实践

    阅读本文大概需要 3 分钟. 大家好,这是[C#.NET 拾遗补漏]专辑的第 06 篇文章.今天讲讲大家熟悉的单例模式. 单例模式大概是所有设计模式中最简单的一种,如果在面试时被问及熟悉哪些设计模式, ...

  4. .net 集合分成几个等数量集合_[C#.NET 拾遗补漏]08:强大的LINQ

    大家好,这是 [C#.NET 拾遗补漏] 系列的第 08 篇文章,今天讲 C# 强大的 LINQ 查询.LINQ 是我最喜欢的 C# 语言特性之一. LINQ 是 Language INtegrate ...

  5. 迭代器、装饰器、软件开发规范

    本节内容 迭代器&生成器 装饰器 Json & pickle 数据序列化 软件目录结构规范 作业:ATM项目开发 1.列表生成式,迭代器&生成器 列表生成式 孩子,我现在有个需 ...

  6. Python之迭代器、装饰器、软件开发规范

    本节内容 迭代器&生成器 装饰器 Json & pickle 数据序列化 软件目录结构规范 作业:ATM项目开发 1.列表生成式,迭代器&生成器 列表生成式 孩子,我现在有个需 ...

  7. 迭代器/生成器/装饰器 /Json pickle 数据序列化

    本节内容 迭代器&生成器 装饰器 Json & pickle 数据序列化 软件目录结构规范 作业:ATM项目开发 1.列表生成式,迭代器&生成器 列表生成式 孩子,我现在有个需 ...

  8. Day4 - Python基础4 迭代器、装饰器、软件开发规范

    Python之路,Day4 - Python基础4 (new版) 本节内容 迭代器&生成器 装饰器 Json & pickle 数据序列化 软件目录结构规范 作业:ATM项目开发 1. ...

  9. Python基础 day4 迭代器生成器 装饰器 Json pickle 数据序列化 软件目录结构规范 作业:ATM项目开发...

    本节内容 迭代器&生成器 装饰器 Json & pickle 数据序列化 软件目录结构规范 作业:ATM项目开发 列表生成器 1.列表生成式,迭代器&生成器 列表生成式 孩子, ...

最新文章

  1. 国外AI教学网红网站
  2. winform 控件半透明设置
  3. python画椭圆turtle_Python turtle画图库画姓名实例
  4. K8s-V1.17.6支持GPU
  5. Excel导出显示服务器意外,C# 调用Excel 出现服务器出现意外状况. (异常来自 HRESULT:0x80010105 (RPC_E_SERVERFAULT)...
  6. cs8900a网卡驱动--寄存器
  7. iss版本服务器读取_【IIS7服务器管理工具下载】IIS7服务器管理 v2.1.9 官方版-开心电玩...
  8. ActiveMQ 无法启动 提示端口被占用 解决方案
  9. python开三次方根函数_Python 中,给 -8 开三次方根出来的是一个虚数,而不是 -2,这怎么办?...
  10. python将图片base流保存为图片文件
  11. 七牛云图片外链失效的解决办法
  12. 一些linux牛皮糖
  13. 如何在 Linux 中查找一个文件
  14. Cocos2d-js cc.director介绍
  15. JavaWeb核心技术系列教程(23)——JSP标签
  16. Catching Cheaters (LCS变形)
  17. 蘑菇街按关键字搜索mogujie商品 API 返回值说明
  18. HTML5特性之谷歌浏览器桌面消息(window.Notification)推送:
  19. 利用Qt来进行文件后缀的更改
  20. 小白入门:python sklearn之kmeans

热门文章

  1. centOS7 安装mysql 设置远程访问
  2. 面向对象——概念(成员变量、静态变量、成员方法、静态方法、垃圾回收机制、重载、包)...
  3. Winform定时启动
  4. [leetcode]347. Top K Frequent Elements
  5. php Collection类的设计
  6. LiveJournal发展历程
  7. excel if in函数_【Excel函数】Small+Index+IF 一对N返回
  8. Mac OS使用技巧之九:Mission Control和DIY自己的Dashboard
  9. Teams Bot 如何使用新的 System.Text.Json 库
  10. 从无到有到完善 - Teams抽奖机器人开发历程