分析CLR Via的方法參數相關
本文通過4各方面對CLR Via進行講解,其中它們有可變數量的參數、方法的參數類型等。希望能給你帶來幫助。
《CLR Via C#》這本書以前就粗略看過兩遍,但一直都沒能深入理解,而且很多內容也忘記了,現在準備重新看一遍,并將看過的部分寫出來,因為寫的過程也是一個加深理解的過程。
本系列算是學習的一個記錄吧,也可以方便以后自己查閱,如果對大家還有些幫助的話,我就很高興了。書我是選擇性的看的,所以順序和書中的順序可能不一樣。
通常我們在使用方法的時候,方法的參數是按值傳遞的,如果傳遞的參數是引用類型的對象,是將引用對象的地址傳給方法。如果傳遞的是值類型實例,傳遞給方法的是該實例的一個副本。CLR Via允許使用在方法中按引用來傳遞參數,C#中使用out和ref關鍵字來體現。下面就來介紹out和ref的使用。
1、CLR Via之out
(1)使用out時在方法的參數定義和方法的調用時都要用out關鍵字,如下:
- static void Main(string[] args)
- {
- string name = string.Empty;
- GetStr(out name); //調用時加out
- Console.WriteLine(name);
- }
- private static void GetStr(out string name) //方法參數定義時的out
- {
- name = "oec2003";
- }
(2)如果一個方法有out修飾的參數,在方法結束前必須給給參數賦值,否則不能通過編譯,代碼如下:
- static void Main(string[] args)
- {
- string name = "oec2003";
- GetStr(out name);
- Console.WriteLine(name);
- }
- private static void GetStr(out string name)
- {
- //沒有給name賦值,編譯時會出現“控制離開當前方法之前必須對 out
- 參數“name”賦值”異常
- }
(3)在調用有out參數的方法時,沒有必要給out參數賦初始值,因為賦的值不會傳遞到方法的內部,如果在方法的內部要強行使用out參數會有編譯錯誤。代碼看下面,通常我們需要在一個方法中返回多個值的時候就可以使用out參數。和out參數一樣,使用ref時在方法的參數定義和方法的調用時都要用ref關鍵字。和out不同的是ref參數的值可以傳入方法內進行操作。
- static void Main(string[] args)
- {
- string name = "oec2003"; //給out參數name賦初始值oec2003
- GetStr(out name);
- Console.WriteLine(name);
- }
- private static void GetStr(out string name)
- {
- name = "hello" + name; //name在調用前雖賦值為oec2003 ,
- //但此處會報錯“使用了未賦值的 out 參數name”
- }
(4)如果兩個方法的參數個數和類型都相同,區別只是其中一個為out參數,那么這兩個方法是可以進行重載的,下面的代碼可以正常運行。
2、CLR Via之ref
- private static void GetStr(out string name)
- {
- name = "oec2003";
- }
- private static void GetStr(string name)
- {
- name = "oec2003";
- }
(1)如果調用方法前,ref參數沒有賦初始值,不能通過編譯,看如下代碼:
- static void Main(string[] args)
- {
- string name;
- GetStr(ref name); //如果name沒有賦值不能通過編譯
- Console.WriteLine(name);
- }
- private static void GetStr(ref string name)
- {
- name = "oec2003";
- }
- static void Main(string[] args)
- {
- string name="oec2003";
- GetStr(ref name);
- Console.WriteLine(name);//返回:hello oc2003
- }
- private static void GetStr(ref string name)
- {
- name = "hello " + name;
- }
(2) 因為ref在傳入方法時會有初始值,所以在方法的內部可以不對ref參數進行任何的操作,那樣ref參數的值不會改變。
- static void Main(string[] args)
- {
- string name="oec2003";
- GetStr(ref name);
- Console.WriteLine(name);//在方法中沒有操作,仍然返回oec2003
- }
- private static void GetStr(ref string name)
- {
- ......
- }
(3)同out參數一樣,如果兩個方法的參數個數和類型都一樣,僅有的區別只是其中之一的參數為ref參數,兩個可以進行重載。
對于CLR Via來說,關鍵字out和關鍵字ref是等價的,就是說無論使用的out還是ref,都會生成相同的IL代碼,正因為如此,如果兩個方法的差異僅僅是out和ref的差異,那么這兩個方法是不能進行重載的,如下代碼:
- //下面代碼編譯會報“不能定義僅在 ref 和 out 上有差別的重載方法”異常
- private static void GetStr(ref string name)
- {
- name = "oec2003";
- }
- private static void GetStr(out string name)
- {
- name = "oec2003";
- }
3、CLR Via之可變數量的參數
有些時候如果一個方法的參數數量可以根據用戶的需要而進行變動,那將會帶來很大的方便。像String類型的Concat、Format等方法就提供了可變參數。可以變參數在C#中使用params來定義,如下面代碼:
- static void Main(string[] args)
- {
- Console.WriteLine(Add(1,2,3,4));
- }
- public static int Add(params int[] num)
- {
- int sum = 0;
- foreach (int i in num)
- {
- sum += i;
- }
- return sum;
- }
使用可變參數非常簡單,需要注意的是可變參數的類型一定要是數組類型。可變參數雖然很好用,但是接受可變參數的方法在調用時會導致一些性能損失,以為數組對象必須分配在堆上,數組的內存最終需要GC來回收。為了避免這種性能的損耗,我們在寫方法的時候可以多定義幾個沒有params關鍵字的方法的重載,這樣只有在很特殊的情況下才會使用有params關鍵字的方法。
4、CLR Via之方法的參數類型
聲明方法參數類型時,應盡可能只用弱的類型,例如如果要編寫一個操作一組數據項的方法,***使用接口(如Ienumerable
- //使用的弱類型參數
- private void OperateCollection<T>(IEnumerable<T> collection)
- {
- }
- //使用的強類型參數
- private void OperateCollection<T>(List<T> collection)
- {
- ......
- }
此處所指的強類型和弱類型,可以理解為類型的層次,如果說父類的層次高于子類,那么層次越高就類型越弱。Iemumerable接口直接在 System.Collections 命名空間下,是其他一些集合類和接口(如Icollection IList List等)的基類,所以定義參數為IEnumerable 類型的,凡是繼承了IEnumerable 的類型的參數都能夠傳入方法,大大提高了靈活性。
以上就是對CLR Via的簡單介紹。
【編輯推薦】