前言:反射(Reflection)是.NET提供給開發者的一個強大工具,儘管做爲.NET框架的使用者,不少時候不會用到反射。但在一些狀況下,尤爲是在開發一些基礎框架或公共類庫時,使用反射會使系統架構更加靈活。數據庫
在開發中,咱們經常會遇到好比有些狀態值在定義好後幾乎從不改動,這時候使用數據庫就顯得有些多餘了。首先想到的一個辦法多是在程序中建立一個數組來表示,此時,咱們遇到了使用數組可能帶來的第一個問題:不方便使用。當數組結構變動時,可能意味着全部使用過此數組的地方的索引都發生了變動,這是咱們不想看到的。數組
這時,咱們能夠採用枚舉:架構
public enum BookingStatus { 未提交 =1, 已提交, 已取消, 已定妥 = 6 }
在實際使用中,可能須要用戶下拉選擇這些狀態值,這時就須要咱們把枚舉綁定到下拉框上(此處以Combobox爲例)了。你們知道,能夠綁定到下拉框列表的有兩種類型:一種是實現了IEnumerable接口的可枚舉類型,好比ArrayList,String[],List<T>;一類是實現了IListSource的數據源,好比DataTable,DataSet。框架
但是,枚舉enum是一個基本類型,它不會實現任何的接口,顯然,沒法直接將枚舉綁定到下拉框上,那麼應該採用什麼辦法呢?工具
使用反射遍歷枚舉字段spa
要遍歷枚舉,首先就須要一個包含枚舉每一個字段信息的對象,這個對象至少應該包含兩個屬性,一個是字段的名稱,一個是字段的值,以方便後續綁定。code
先看下完整的代碼:orm
private void Form1_Load(object sender, EventArgs e) { comboBox1.DataSource = GetDataTable(); comboBox1.DisplayMember = "Name"; comboBox1.ValueMember = "Value"; } public static DataTable GetDataTable() { Type t = typeof(BookingStatus); //建立類型 FieldInfo[] fieldinfo = t.GetFields(); //獲取字段信息對象集合 DataTable table = new DataTable(); table.Columns.Add("Name", typeof(String)); table.Columns.Add("Value", typeof(Int32)); foreach (FieldInfo field in fieldinfo) { if (!field.IsSpecialName) { DataRow row = table.NewRow(); row[0] = field.Name; //獲取文本字段 row[1] = (int)field.GetRawConstantValue(); //獲取int數值 table.Rows.Add(row); } } return table; } } public enum BookingStatus { 未提交 =1, 已提交, 已取消, 已定妥 = 6 }
效果:對象
這裏大體作一個說明:Type抽象類提供了訪問類型元數據的能力,當實例化了一個Type對象後,能夠經過它的屬性和方法,獲取類型的元數據的信息,或者進一步得到該類型的成員的元數據信息。注意到這裏,由於Type對象老是基於某一具體類型的,而且它是一個抽象類,因此再建立Type類型時,須要提供類型名稱或者類型的實例。程序集元數據,經過Type類型就能夠訪問類型的元數據信息,而訪問類型元數據的操做,就叫作反射。blog
使用泛型來達到代碼重用
觀察上面的代碼,若是如今有另外一個枚舉,叫作TicketStatus,那麼要將它的枚舉項文本和值轉換爲DataTable,惟一須要改動的就是這裏:
Type t = typeof(BookingStatus); //將枚舉名稱更換
既然這樣,就能夠利用泛型來實現代碼的重用,將返回DataTable的方法更改成泛型方法:
public static DataTable GetDataTable<T>() { Type t = typeof(T); //建立類型 FieldInfo[] fieldinfo = t.GetFields(); //獲取字段信息對象集合 //...............省略 return table; }
從以上代碼能夠看出,綜合使用反射,泛型等技術,能夠極大地提升代碼的靈活性,可重用性。