新聞中心
這個(gè)系列,我和大家一起來(lái)學(xué)學(xué)C#中一些非常有用函數(shù),對(duì)于有些人來(lái)說可能它們不起眼,因此常常忽略它們。它們不會(huì)嚷嚷到:“使用我吧!我會(huì)讓你節(jié)省很多的時(shí)間,而且讓你的代碼變得更加的簡(jiǎn)潔!” -- 但是,這些話會(huì)從對(duì)它們熟悉的程序員的腦海中涌現(xiàn)出來(lái)。

好的!先來(lái)看看ToLookup:
故事的背景
讓我們先來(lái)創(chuàng)建一個(gè)簡(jiǎn)單的類來(lái)表示產(chǎn)品,產(chǎn)品有ID,類別,和價(jià)格,這個(gè)類沒有什么特別:
- public sealed class Product
- public int Id { get; set; }
- public string Category { get; set; }
- public double Value { get; set; }
- public override string ToString()
- {
- return string.Format("[{0}: {1} - {2}]", Id, Category, Value);
- }
然后我們加入一個(gè)函數(shù)得到一個(gè)產(chǎn)品的列表,當(dāng)然你也可以從數(shù)據(jù)庫(kù)中讀取出來(lái):
- public static List
GetList() - {
- var products = new List
- {
- new Product {Id = 1, Category = "Electronics", Value = 15.0},
- new Product {Id = 2, Category = "Groceries", Value = 40.0},
- new Product {Id = 3, Category = "Garden", Value = 210.3},
- new Product {Id = 4, Category = "Pets", Value = 2.1},
- new Product {Id = 5, Category = "Electronics", Value = 19.95},
- new Product {Id = 6, Category = "Pets", Value = 21.25},
- new Product {Id = 7, Category = "Pets", Value = 5.50},
- new Product {Id = 8, Category = "Garden", Value = 13.0},
- new Product {Id = 9, Category = "Automotive", Value = 10.0},
- new Product {Id = 10, Category = "Electronics", Value = 250.0},
- };
- return products;
- }
我們有一個(gè)任務(wù)就是按類別列出一個(gè)物品清單,這個(gè)非常的容易,用GroupBy 就可以實(shí)現(xiàn)了:
- foreach (var group in products.GroupBy(p => p.Category))
- Console.WriteLine(group.Key);
- foreach (var item in group)
- {
- Console.WriteLine("\t" + item);
- }
看起來(lái)一切都很好,沒有什么問題.
當(dāng)我們使用 GroupBy() 擴(kuò)展方法時(shí),使用了延遲執(zhí)行。 這意味著,當(dāng)你遍歷集合的時(shí)候,下一個(gè)要出現(xiàn)的項(xiàng)目可能會(huì)或者可能不會(huì)被加載。 這是一個(gè)很大的性能改進(jìn),但它會(huì)引起有趣的副作用。
在用 GroupBy()時(shí), 它實(shí)際上是在***項(xiàng)被使用的時(shí)候創(chuàng)建分組,而不是在 GroupBy() ***次被調(diào)用時(shí)。
考慮一下:如果你從數(shù)據(jù)庫(kù)中加載數(shù)據(jù),然后想組合到一起,并存儲(chǔ)快速查找。 看下面的一段代碼:
- var groups = products.GroupBy(p => p.Category);
- //刪除所有屬于Garden的產(chǎn)品
- products.RemoveAll(p => p.Category == "Garden");
- foreach (var group in groups)
- {
- Console.WriteLine(group.Key);
- foreach (var item in group)
- {
- Console.WriteLine("\t" + item);
- }
- }
執(zhí)行后發(fā)現(xiàn),所有的Garden產(chǎn)品都已經(jīng)消失了,但是 groups 是在執(zhí)行刪除命令前就已經(jīng)賦值了。
基于這種情況,我們不得不使用ToDictionary() 將GroupBy 后的結(jié)果儲(chǔ)存起來(lái),這樣一來(lái)工作量就增加了,而且維護(hù)也不太方便 -- 請(qǐng)大家試試。
ToLookup登場(chǎng)
現(xiàn)在我們有請(qǐng)ToLookup。
該 ToLookup() 方法創(chuàng)建一個(gè)類似 字典(Dictionary ) 的列表List, 但是它是一個(gè)新的 .NET Collection 叫做 lookup。 Lookup,不像Dictionary, 是不可改變的。 這意味著一旦你創(chuàng)建一個(gè)lookup, 你不能添加或刪除元素。
- var productsByCategory = products.ToLookup(p => p.Category);
- foreach (var group in productsByCategory)
- {
- // the key of the lookup is in key property
- Console.WriteLine(group.Key);
- // the list of values is the group itself.
- foreach (var item in group)
- {
- Console.WriteLine("\t" + item);
- }
- }
你還可以使用類似索引的功能得到某個(gè)項(xiàng)目,在本案例中是得到某個(gè)類別的所有產(chǎn)品:
- private static void PrintCategory(ILookup
productsByCategory,string categoryName) - {
- foreach (var item in productsByCategory[categoryName])
- {
- Console.WriteLine(item);
- }
- }
結(jié)論
ToLookup() 是一個(gè)奇妙的函數(shù),用于對(duì)一個(gè)集合進(jìn)行操作,創(chuàng)建一個(gè)1:n 的映射。 它可以方便的將數(shù)據(jù)分類成組,并生成一個(gè)字典供查詢使用。
新聞標(biāo)題:走進(jìn)C#奇妙函數(shù)之ToLookup
網(wǎng)址分享:http://m.fisionsoft.com.cn/article/dhpsoji.html


咨詢
建站咨詢
