看來Cast<T>好像是為 Control 准備,Control 類和Control 的派生類多處使用了非泛型。
可現在都用vs2008(甚至vs2010)了,那為什麼WinForm的窗體控件還用非泛型,太落後了吧!!!
確實如此,WinForm對泛型控件(Control)的支持上存在很大問題。
雖然可以定義泛型控件,也可以使用,可以運行。但會有很多麻煩的,比如窗體設計器沒法顯示...
那只好使用非泛型的了,好在我們有Cast<T>!
再來看看Cast<T>對繼承的支持,我們定義兩個類A和B,B繼承自A,如下:
1 public class A { }
2 public class B : A { }
來試試如下類型轉換操作:
1 //子類集合
2 B[] bb = new B[] { new B(), new B(), new B(), new B() };
3 //轉換成父類
4 A[] aa = bb.Cast<A>().ToArray();
5 //再轉回子類
6 B[] bb2 = aa.Cast<B>().ToArray();
以上三個操作都可編譯並運行通過,修改下再試:
1 A[] aa = new A[] { new A(), new A(), new B() };
2 B[] bb3 = aa2.Cast<B>().ToArray();
這次不行了,將父類cast為子類可不是隨意的:
不過我們有解決辦法,我們使用Enumerable.OfType<T>,是Cast<T>的親兄弟,如下使用:
1 B[] bb = aa.OfType<B>().ToArray();
看了上面的,總感覺Cast<T>的內部只是執行了(T)enumerator.Current這樣一個簡單操作,讓我們再用 int 和 double 轉換驗證一下:
1 int i = (int)1.001;
2 double d = (double)10;
3
4 int[] ints1 = new int[] { 1, 2, 3, 4, 5 };
5 double[] ds1 = ints1.Cast<double>().ToArray();
6
7 double[] nums1 = new double[] { 1.0001, 2.0003, 3.001, 3.9997, 4.002 };
8 int[] nums2 = nums1.Cast<int>().ToArray();
1、2行為強制類型轉換,沒問題。(當然第2行的(double)可以省略。)
第 5 行試圖將整數集合轉換為double集合,運行時會報錯:
第7行也會報同樣的錯誤。看來Cast<T>內部並非只是簡單轉換!
用Reflect反編譯了一下,用到了下面這個類:
反編譯後代碼比較亂,加上本人水平有限,也沒弄明白,還是把這個難題留給園子裡的高手吧!
總結:
1. Cast<T>可廣泛應用在WinForm的控件上;
2. 有類繼承的集合轉換上,建議用OfType<T>;
3. Cast<T>不能理解成簡單類型轉換。