Enumerable.Cast的用途討論
Enumerable.Cast< T>用于將IEnumerable轉(zhuǎn)換為泛型版本IEnumerable< T>。轉(zhuǎn)換后可盡情享用Enumerable的其它方法(如Where、Select),給我們的編碼帶來極大便利。
但MSDN中僅給出一個轉(zhuǎn)換ArrayList的例子,很多人看了感覺現(xiàn)在都在用List< T>,還有誰會用ArrayList,Cast< T>沒多少用處,除非處理一些之前遺留的一些代碼。
其實Cast< T>并非如此簡單,它可以用在很多地方。
Enumerable.Cast< T>使用范例
先看MSDN中舉的例子吧:
- System.Collections.ArrayList fruits = new System.Collections.ArrayList();
- fruits.Add("apple");
- fruits.Add("mango");
- IEnumerable< string> query = fruits.Cast< string>();
- foreach (string fruit in query) Console.WriteLine(fruit);
這個例子比較簡單,很容易理解。
同樣.Net 1.x中的其它幾個集合類也可如上使用,如Array、非泛型版的List...
打斷,有沒有非泛型版的List?我沒太用過.Net 1.x,不太清楚,不過窗體控件中是有個List控件(ASP.Net)和一個ListView控件(WinForm)。
就以ListView為例子吧,ListView控件可以包含很多項,也可以說是一個集合,就讓我們來看看它的Items屬性吧!
- public class ListView : Control
- {
- public ListView.ListViewItemCollection Items { get; }
- public class ListViewItemCollection : IList, ICollection, IEnumerable { }
- }
ListView的Items類型是ListView.ListViewItemCollection,這個ListViewItemCollection實現(xiàn)了IEnumerable。
ListView.Items正是一個非泛型的集合,因此可以應用Cast< T>。
以下代碼假定 listBox 數(shù)據(jù)綁定在一個Employee的集合上:
- int count = listBox.Items.Cast< Employee>().Count();
- bool b = listBox.Items.Cast< Employee>().Any(e => e.FirstName == "Bob");
(當然,如果有Employee的集合的引用,就不用Cast了,這里只是示例)
同樣Cast< T>可以用在ComboBox、DataGridView、TreeNode上:
- //ComboBox
- var v1 = comboBox.Items.Cast< People>();
- //DataGridView
- var v2 = dataGridView.SelectedRows.Cast< DataGridViewRow>();
- var v3 = dataGridView.SelectedColumns.Cast< DataGridViewColumn>();
- var v4 = dataGridView.SelectedCells.Cast< DataGridViewCell>();
- //TreeNode
- var v5 = treeNode.Nodes.Cast< TreeNode>();
這幾個應用中應該第 4 行的應用最多,獲取選中行是DataGridView使用最頻繁的操作之一。
試看下面代碼:
- //計算平均年齡
- int age = dataGridView.SelectedRows.Cast< Employee>().Average(p=>p.Age);
- //統(tǒng)計所在城市
- string[] cities = dataGridView.SelectedRows.Cast< Employee>().Select(p => p.City).Distinct();
用了Cast< T>,我們的代碼很精簡。
Enumerable.Cast< T>使用在基類Control上
Cast< T>甚至還可以用在所有控件的基類Control上,它的Controls屬性也是非泛型的!
- //Control
- var v6 = control.Controls.Cast< Control>();
看來Cast< T>好像是為 Control 準備,Control 類和Control 的派生類多處使用了非泛型。
可現(xiàn)在都用vs2008(甚至vs2010)了,那為什么WinForm的窗體控件還用非泛型,太落后了吧!??!
確實如此,WinForm對泛型控件(Control)的支持上存在很大問題。
雖然可以定義泛型控件,也可以使用,可以運行。但會有很多麻煩的,比如窗體設(shè)計器沒法顯示...
那只好使用非泛型的了,好在我們有Cast< T>!
再來看看Cast< T>對繼承的支持,我們定義兩個類A和B,B繼承自A,如下:
- public class A { }
- public class B : A { }
來試試如下類型轉(zhuǎn)換操作:
- //子類集合
- B[] bb = new B[] { new B(), new B(), new B(), new B() };
- //轉(zhuǎn)換成父類
- A[] aa = bb.Cast< A>().ToArray();
- //再轉(zhuǎn)回子類
- B[] bb2 = aa.Cast< B>().ToArray();
以上三個操作都可編譯并運行通過,修改下再試:
- A[] aa = new A[] { new A(), new A(), new B() };
- B[] bb3 = aa2.Cast< B>().ToArray();
這次不行了,將父類cast為子類可不是隨意的:
不過我們有解決辦法,我們使用Enumerable.OfType< T>,是Cast< T>的親兄弟,如下使用:
- B[] bb = aa.OfType< B>().ToArray();
看了上面的,總感覺Cast< T>的內(nèi)部只是執(zhí)行了(T)enumerator.Current這樣一個簡單操作,讓我們再用 int 和 double 轉(zhuǎn)換驗證一下:
- int i = (int)1.001;
- double d = (double)10;
- int[] ints1 = new int[] { 1, 2, 3, 4, 5 };
- double[] ds1 = ints1.Cast< double>().ToArray();
- double[] nums1 = new double[] { 1.0001, 2.0003, 3.001, 3.9997, 4.002 };
- int[] nums2 = nums1.Cast< int>().ToArray();
1、2行為強制類型轉(zhuǎn)換,沒問題。(當然第2行的(double)可以省略。)
第 5 行試圖將整數(shù)集合轉(zhuǎn)換為double集合,運行時會報錯:
第7行也會報同樣的錯誤??磥鞢ast< T>內(nèi)部并非只是簡單轉(zhuǎn)換!
用Reflect反編譯了一下,用到了下面這個類:
反編譯后代碼比較亂,加上本人水平有限,也沒弄明白,還是把這個難題留給園子里的高手吧!
Enumerable.Cast< T>總結(jié):
1. Cast< T>可廣泛應用在WinForm的控件上;
2. 有類繼承的集合轉(zhuǎn)換上,建議用OfType< T>;
3. Cast< T>不能理解成簡單類型轉(zhuǎn)換。
希望:
1. 大家把自己知道的Cast< T>的擴展寫在回復中,分享給大家。
2. 高手們能給出Cast< T>內(nèi)部實現(xiàn)的簡單描述,讓大家用的明明白白。
本文來自鶴沖天博客:《c#擴展方法奇思妙用》
【編輯推薦】