宣告安全型別的清單(list)
安全型別讓撰寫出來的程式碼更容易維護。安全型別的語言在編譯時期就找出bug,而不是在執行時期才被發現。在C# 1.X,宣告的general collection(stack, list)是可以接受任何型別的變數。
在1.X版的Framework中,collection所存放的實體為System.Object,且任何變數都源自於System.Object,所以collection可以取得任何型别的變數,換句話說,它並不安全。
假如想宣告一個存放Employee物件的list,在C# 1.X中,會使用到存放System.Object物件的ArrayList。將Employee物件加入到一個collection並不是難事,因為Employee本身就是從System.Object所衍生。如果想從ArrayList中取出Employee物件,只能取得一個Object的參照,如同下面的程式碼:
Employee theEmployee = (Employee) myArrayList[1];
還有一個更大的問題,沒有一個機制可以阻止把字串或其它型別的物件加入ArrayList中,如果你從沒想過要取得字串,你也不會去注意這個錯誤的型別。假設今天透過方法(method),你希望存入Employee至ArrayList中,當method在執行時期試圖存入字串並轉成Employee型別,就會產生「例外」。
最後一個問題,就是在.Net 1.X的collection中,加入數值型別的變數也會發生這個問題,數值型別會照前面的方法裝箱(be boxed),也會被同樣的方法提取(unboxed)
在.NET 2.0的新函式庫:System.Collection.Generic命名空間已經解決了這個問題。Generic collection變得更容易使用且能讓你指定型別。宣告時,編譯器只會允許一種型別加入list中。定義一個generic collections需使用一個特別的語法;這個語法使用一個角型括孤來表示。
在這裡你不需要去思索從collection取回物件這個問題,因為你的code比起無型別的ArrayList,變得更安全、更容易去維護,而且更容易去使用。
我該怎麼作?
現在,打開你的Visual Studio 2005,並照著下面的範例Example 1-1作。
http://www.dyu.edu.tw/~f9106007/CSharp/CreateATypeSafeList.rar
發生了什麼事?
範例程式產生了兩個類別,Employee:負責處理collection,Program:由Visual Studio 2005所產生出來的。同樣地也是使用由.NET Framework提供的List類別。
Employee類別包含一個empID、建構以及經過覆寫(override)以用來回傳empID字串的ToString方法。
首先宣告一個存放Employee的list實體empList,宣告方式如下:
List<Employee> empList
List<T>這裡,T所表示的意思是目前宣告的List,所存放資料的型別為何。
其實,empList只是一個參照到以new指令產生實體在堆積上(heap)的變數。new指令後面搭配建構子呼叫,如下列所示:
new List<Employee>()
這個動作會產生「存放Employee實體的list」的實體(存在於堆積),你也可以將前述的兩個動作合在一起。
List<Employee> empList = new List<Employee>();
接下來宣告第二個List,型別為「存放整數的List」:
List<int> intList = new List<int>();
現在你可以任意的將整數加進intList或是把Employee加進empList裡。當list裡有了數筆資料,可以利用 foreach來檢視裡面的所有資料:
foreach(Employee employee in empList)
{
Console.Write(“{0}”, employee.ToString());
}
關於…
如果試著將整數存進型別為Employee的list,會發生什麼事呢?將之前的程式稍作修改並重新編譯:
empList.Add(i*5);
接下來會有兩個錯誤產生:
錯誤 1 最符合的多載方法 'System.Collections.Generic.List<CreateATypeSafeList.Employee>.Add(CreateATypeSafeList.Employee)' 有一些無效的引數
錯誤 2 引數 '1': 無法從 'int' 轉換為 'CreateATypeSafeList.Employee'
上面的錯誤告訴我們不能將int的值加入型別為Employee的collection中,因為這其中的轉換是無效的。