Linq延時執(zhí)行全面分析
本文向大家介紹Linq延時執(zhí)行,可能好多人還不了解Linq延時執(zhí)行,沒有關系,看完本文你肯定有不少收獲,希望本文能教會你更多東西。
Linq的大多數(shù)查詢運算符的一個重要特性就是,他們并不是在構建的時候就立即執(zhí)行,而是在枚舉是執(zhí)行,換句話說,當枚舉變量調用MoveNext時執(zhí)行。
在構建查詢之后,另外插入到列表中的數(shù)字也會包含在結構中,因為直到foreach運行時此才回進行篩選或者是排序操作,稱之為延時執(zhí)行或延緩計算,所有標準查詢運算符均為延時執(zhí)行,但是有的運算符不支持延時執(zhí)行的機制,而是立即執(zhí)行,如Count 、ToAarry、toLookup等。
Linq延時執(zhí)行還有一點不好的影響,如果查詢的lambda表達式引用了局部變量,那么這些便來那個會受到外部便來那個語義的約束。
當在foreach循環(huán)中構建查詢時,這就會成為一個陷阱,例如假定想要刪除字符串中的所有元音字母。如下所示的雖然效率不高,但是能得到正確的結果:
- 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
 
上面的代碼是可以得到正確的結果,那么現(xiàn)在如果用foreach循環(huán)重寫這段代碼能否的道正確的結果:
- 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',這是因為編譯器將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);
 - }
 
因為vowel變量時循環(huán)外部聲明的,同一個變量重復聲明更新,所以每個lambda表達式獲取的是同樣的vowel.之后枚舉查詢時,所有的lambda表達式引用了這個變量的當前值,即'u'。為了解決這個問題,必須將循環(huán)變量賦值被另一再循環(huán)代碼塊內聲明的變量:
- 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延時執(zhí)行。
【編輯推薦】















 
 
 
 
 
 
 