深入了解c# 迭代器和列舉器
大家好,這是 [C#.NET 拾遺補(bǔ)漏] 系列的第 07 篇文章。
在 C# 中,大多數(shù)方法都是通過(guò) return 語(yǔ)句立即把程序的控制權(quán)交回給調(diào)用者,同時(shí)也會(huì)把方法內(nèi)的本地資源釋放掉。而包含 yield 語(yǔ)句的方法則允許在依次返回多個(gè)值給調(diào)用者的期間保留本地資源,等所有值都返回結(jié)束時(shí)再釋放掉本來(lái)資源,這些返回的值形成一組序列被調(diào)用者使用。在 C# 中,這種包含 yield 語(yǔ)句的方法、屬性或索引器就是迭代器。
迭代器中的 yield 語(yǔ)句分為兩種:
- yeild return,把程序控制權(quán)交回調(diào)用者并保留本地狀態(tài),調(diào)用者拿到返回的值繼續(xù)往后執(zhí)行。
- yeild break,用于告訴程序當(dāng)前序列已經(jīng)結(jié)束,相當(dāng)于正常代碼塊的 return 語(yǔ)句(迭代器中直接使用 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
實(shí)際場(chǎng)景中,我們一般很少直接寫(xiě)迭代器,因?yàn)榇蟛糠中枰膱?chǎng)景都是數(shù)組、集合和列表,而這些類型內(nèi)部已經(jīng)封裝好了所需的迭代器。比如 C# 中的數(shù)組之所以可以被遍歷是因?yàn)樗鼘?shí)現(xiàn)了 IEnumerable 接口,通過(guò) GetEnumerator() 方法可以獲得數(shù)組的列舉器 Enumerator,而該列舉器就是通過(guò)迭代器來(lái)實(shí)現(xiàn)的。比如最常見(jiàn)的一種使用場(chǎng)景就是遍歷數(shù)組中的每一個(gè)元素,如下面逐個(gè)打印數(shù)組元素的示例。
int[] numbers = { 1, 2, 3, 4, 5 };
IEnumerator enumerator = numbers.GetEnumerator();
while (enumerator.MoveNext())
{
Console.WriteLine(enumerator.Current);
}
其實(shí)這就是 foreach 的工作原理,上面代碼可以用 foreach 改寫(xiě)如下:
int[] numbers = { 1, 2, 3, 4, 5 };
foreach (int number in numbers)
{
Console.WriteLine(number);
}
當(dāng)然,列舉器不一定非要通過(guò)迭代器實(shí)現(xiàn),例如下面這個(gè)自定義的列舉器 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);
}
}
理解迭代器和列舉器可以幫助我們寫(xiě)出更高效的代碼。比如判斷一個(gè) IEnumerable<T> 對(duì)象是否包含元素,經(jīng)??吹接行┤诉@么寫(xiě):
if(enumerable.Count() > 0)
{
// 集合中有元素
}
但如果用列舉器的思維稍微思考一下就知道,Count() 為了獲得集合元素?cái)?shù)量必然要迭代完所有元素,時(shí)間復(fù)雜度為 O(n)。而僅僅是要知道集合中是否包含元素,其實(shí)迭代一次就可以了。所以效率更好的做法是:
if(enumerable.GetEnumerator().MoveNext())
{
// 集合中有元素
}
這樣寫(xiě)時(shí)間復(fù)雜度是 O(1),效率顯然更高。為了書(shū)寫(xiě)方便,C# 提供了擴(kuò)展方法 Any()。
if(enumerable.Any())
{
// 集合中有元素
}
所以如有需要,應(yīng)盡可能使用 Any 方法,效率更高。
再比如在 EF Core 中,需要執(zhí)行 IQueryable<T> 查詢時(shí),有時(shí)候使用 AsEnumerable() 比使用 ToList、ToArray 等更高效,因?yàn)?ToList、ToArray 等會(huì)立即執(zhí)行列舉操作,而 AsEnumerable() 可以把列舉操作延遲到真正被需要的時(shí)候再執(zhí)行。當(dāng)然也要考慮實(shí)際應(yīng)用場(chǎng)景,Array、List 等更方便調(diào)用者使用,特別是要獲取元素總數(shù)量、增刪元素等這種操作。
以上就是深入了解c# 迭代器和列舉器的詳細(xì)內(nèi)容,更多關(guān)于c# 迭代器和列舉器的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C#實(shí)現(xiàn)ArrayList動(dòng)態(tài)數(shù)組的示例
ArrayList是一個(gè)動(dòng)態(tài)數(shù)組,可以用來(lái)存儲(chǔ)任意類型的元素,本文就來(lái)介紹一下C#實(shí)現(xiàn)ArrayList動(dòng)態(tài)數(shù)組的示例,具有一定的參考價(jià)值,感興趣的可以了解一下2023-12-12
.NET連接MongoDB數(shù)據(jù)庫(kù)實(shí)例教程
這則小竅門(mén)將講述如何開(kāi)發(fā)一個(gè).NET應(yīng)用來(lái)連接Mongo數(shù)據(jù)庫(kù)并執(zhí)行多種操作。同時(shí)還稍微涉及了Mongo數(shù)據(jù)庫(kù)和多種命令2013-11-11
C#實(shí)現(xiàn)簡(jiǎn)單的RSA非對(duì)稱加密算法示例
這篇文章主要介紹了C#實(shí)現(xiàn)簡(jiǎn)單的RSA非對(duì)稱加密算法,結(jié)合實(shí)例形式分析了C#實(shí)現(xiàn)RSA加密的具體步驟與相關(guān)操作技巧,需要的朋友可以參考下2017-08-08
微信跳一跳自動(dòng)腳本C#代碼實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了微信跳一跳自動(dòng)腳本C#代碼實(shí)現(xiàn)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01
Stream.Write 與 StreamWriter.Write 的不同
Stream.Write 與 StreamWriter.Write 是我們?cè)谙蛄髦袑?xiě)數(shù)據(jù)時(shí),最常用的方法。下面就詳細(xì)講解這兩個(gè)方法。2013-04-04
C# DataGridView綁定數(shù)據(jù)源的方法
這篇文章主要為大家詳細(xì)介紹了C# DataGridView綁定數(shù)據(jù)源的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-09-09

