Linq延時(shí)執(zhí)行全面分析
本文向大家介紹Linq延時(shí)執(zhí)行,可能好多人還不了解Linq延時(shí)執(zhí)行,沒(méi)有關(guān)系,看完本文你肯定有不少收獲,希望本文能教會(huì)你更多東西。
Linq的大多數(shù)查詢運(yùn)算符的一個(gè)重要特性就是,他們并不是在構(gòu)建的時(shí)候就立即執(zhí)行,而是在枚舉是執(zhí)行,換句話說(shuō),當(dāng)枚舉變量調(diào)用MoveNext時(shí)執(zhí)行。
在構(gòu)建查詢之后,另外插入到列表中的數(shù)字也會(huì)包含在結(jié)構(gòu)中,因?yàn)橹钡絝oreach運(yùn)行時(shí)此才回進(jìn)行篩選或者是排序操作,稱之為延時(shí)執(zhí)行或延緩計(jì)算,所有標(biāo)準(zhǔn)查詢運(yùn)算符均為延時(shí)執(zhí)行,但是有的運(yùn)算符不支持延時(shí)執(zhí)行的機(jī)制,而是立即執(zhí)行,如Count 、ToAarry、toLookup等。
Linq延時(shí)執(zhí)行還有一點(diǎn)不好的影響,如果查詢的lambda表達(dá)式引用了局部變量,那么這些便來(lái)那個(gè)會(huì)受到外部便來(lái)那個(gè)語(yǔ)義的約束。
當(dāng)在foreach循環(huán)中構(gòu)建查詢時(shí),這就會(huì)成為一個(gè)陷阱,例如假定想要?jiǎng)h除字符串中的所有元音字母。如下所示的雖然效率不高,但是能得到正確的結(jié)果:
- IEnumberable<int> qurey ="Not what you might expect";
- queryquery =query.Where(c=>c!='a');
- queryquery =query.Where(c=>c!='e');
- queryquery =query.Where(c=>c!='i');
- queryquery =query.Where(c=>c!='o');
- queryquery =query.Where(c=>c!='u');
- foreach(char c in query)
- Console.Write(c);
- //Nt wht y mght xpct
上面的代碼是可以得到正確的結(jié)果,那么現(xiàn)在如果用foreach循環(huán)重寫這段代碼能否的道正確的結(jié)果:
- IEnumerable<char> query ="Not what you might expect";
- foreach(cha vowel in "aeiou")
- qwuer =query.Where(c=>c!=vowel);
- foreach(char c in query)
- Console.Write(c); //Not what yo might expect
- //只刪除了'u',這是因?yàn)榫幾g器將foreach循環(huán)翻譯成如下的代碼:
- IEnumerable<char> vowels="aeiou";
- Iemuberator<char> rator=vowels.GetEnumerator();
- char vowel;
- while(rator.MoveNext())
- {
- vowel =rator.Current;
- queryquery =query.Where(c=>c!=vowel);
- }
因?yàn)関owel變量時(shí)循環(huán)外部聲明的,同一個(gè)變量重復(fù)聲明更新,所以每個(gè)lambda表達(dá)式獲取的是同樣的vowel.之后枚舉查詢時(shí),所有的lambda表達(dá)式引用了這個(gè)變量的當(dāng)前值,即'u'。為了解決這個(gè)問(wèn)題,必須將循環(huán)變量賦值被另一再循環(huán)代碼塊內(nèi)聲明的變量:
- IEnumberable<int> qurey ="Not what you might expect";
- foreach( char vowel in "aeiou")
- {
- char temp =vowel;
- queryquery =query.Where(c=>c!=temp);
- }
這樣每次循環(huán)迭代守使用新的變量,以上介紹Linq延時(shí)執(zhí)行。
【編輯推薦】