深入了解c# 迭代器和列举器

作者:精致码农 • 王亮 时间:2022-04-24 16:32:37 

大家好,这是 [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 等更方便调用者使用,特别是要获取元素总数量、增删元素等这种操作。

来源:https://www.cnblogs.com/willick/p/13473526.html

标签:c#,迭代器,列举器
0
投稿

猜你喜欢

  • 详解Java中NullPointerException异常的原因详解以及解决方法

    2023-01-22 20:13:28
  • Spring中自动装配的4种方式

    2021-08-01 16:44:41
  • 详解Java中多进程编程的实现

    2021-12-22 01:46:06
  • Hibernate三种状态和Session常用的方法

    2021-11-28 15:04:43
  • 一文梳理Java 8后的新功能

    2023-07-13 07:41:43
  • 使用Stargate访问K8ssandra的过程之Springboot整合Cassandra

    2022-02-08 23:12:25
  • 你真的知道Java中对象的销毁吗

    2022-09-14 22:31:38
  • RocketMQ消息生产者是如何选择Broker示例详解

    2023-11-10 21:45:49
  • 利用Java8 Optional类优雅如何地解决空指针问题

    2023-07-30 04:58:13
  • C语言运算符优先级列表(超详细)

    2023-07-04 08:17:54
  • java使用计算md5校验码方式比较两个文件是否相同

    2023-05-10 23:43:16
  • 最常用的1000个Java类(附代码示例)

    2023-03-25 20:29:07
  • java中Hashmap的get方法使用

    2023-10-29 13:10:05
  • androidQ sd卡权限使用详解

    2021-09-27 17:38:51
  • 解决nacos升级spring cloud 2020.0无法使用bootstrap.yml的问题

    2021-12-02 19:44:29
  • SpringBoot项目的logback日志配置(包括打印mybatis的sql语句)

    2021-08-24 02:15:22
  • java 交换两个数据的方法实例详解

    2021-12-06 00:56:04
  • Maven+Tomcat8 实现自动化部署的方法

    2023-01-03 06:44:20
  • SpringBoot 嵌入式web容器的启动原理详解

    2021-12-29 23:23:14
  • springcloud-gateway整合jwt+jcasbin实现权限控制的详细过程

    2023-11-20 12:57:09
  • asp之家 软件编程 m.aspxhome.com