C# 9 新特性 —— 增強的 foreach
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
~~