Intro
在 C# 9 中增强了 foreach 的使用,使得一切对象都有 foreach 的可能
我们来看一段代码,这里我们试图遍历一个 int 类型的值
思考一下,我们可以怎么做使得上面的代码编译通过呢?
迭代器模式
迭代器模式,提供一种方法顺序访问一个聚合对象中的各个元素,而又不暴露该对象的内部表示。
迭代器模式是分离了集合对象的遍历行为,抽象出一个迭代器类来负责,这样既可以做到不暴露集合的内部结构,又可以让外部代码透明地访问集合内部的数据。
foreach 其实是一个迭代器模式的语法糖,用来遍历一个集合中的数据,foreach 可以使用 while 来实现,比如下面这个示例:
var enumerable = Enumerable.Range(1, 10).ToArray(); foreach (var i in enumerable) { Console.WriteLine(i); }
使用 while 重写之后类似下面这样的代码:
var enumerator = enumerable.GetEnumerator(); while (enumerator.MoveNext()) { Console.WriteLine(enumerator.Current); }
c# 中的集合基本都实现了迭代器模式,可以直接使用 foreach 来遍历,对于自定义的类型想要支持 foreach 可以实现 IEnumerable 或 IEnumerable<T>,对于没有实现迭代器的代码,是不是可以用 foreach 呢
Enumerator
我们再来看开篇提到的问题,怎么实现支持 foreach 呢
从上面 VS 的提示我们可以看得出来,如果一个类型想要支持 foreach,有三种方式可以实现:
- 实现 IEnumerable
- 实现 IEnmuerable<T>
- 添加 GetEnumerator 方法,方法返回值类型需要有 Current 属性和 MoveNext 方法,可以参考这个 IEnumerator,返回类型可以直接实现 IEnumerator 或 IEnumerator<T>
那么如果是一个别人封装的类型,能否支持 foreach 呢,从 C# 9 之后就可以了,可以添加一个 GetEnumerator 的扩展方法,类似于下面
public static class ForEachExtensions { public static IEnumerator<char> GetEnumerator(this int num) { return num.ToString().GetEnumerator(); } }
此时如果是使用 C# 9 就可以编译通过了,如果手动设置了 LangVersion,需要修改为 9,否则会得到类似下面这样的错误
添加使用扩展方法,并启用 C# 9 语法:
More
有了这个功能之后,一切类型都是可以 foreach 的,没有实现迭代器模式的类型,只需要实现一个扩展方法就可以了
迎接 C# 9 ,万物皆可 foreach ~~
Reference
- https://docs.microsoft.com/en-us/dotnet/csharp/whats-new/csharp-9
- https://github.com/WeihanLi/SamplesInPractice/tree/master/CSharp9Sample
- https://github.com/WeihanLi/SamplesInPractice/blob/master/CSharp9Sample/ForEachExtensions.cs
到此这篇关于C# 9 新特性之增强的foreach的文章就介绍到这了,更多相关C#9新特性之foreach内容请搜索自由互联以前的文章或继续浏览下面的相关文章希望大家以后多多支持自由互联!