一、引言
對於一些初學者(包括工作幾年的人在內)來說,有時候對於方法之間的參數傳遞的問題感覺比較困惑的,因為之前在面試的過程也經常遇到參數傳遞的基礎面試題,這樣的面試題主要考察的開發人員基礎是否扎實,對於C#中值類型和引用類型有沒有深入的一個理解——這個說的理解並不是簡單的對它們簡單一個定義描述,而在於它們在內存中分布。所以本文章將帶領大家深入剖析下C#中參數傳遞的問題,並分享我自己的一個理解,只有你深入理解了才能在不運行程序的情況就可以分析出參數傳遞的結果的。
二、按值傳遞
對於C#中的參數傳遞,根據參數的類型可以分為四類:
- 值類型參數的按值傳遞
- 引用類型參數的按值傳遞
- 值類型參數的按引用傳遞
- 引用類型參數的按引用傳遞
然而在默認情況下,CLR方法中參數的傳遞都是按值傳遞的。為了幫助大家全面理解參數的傳遞,下面就這四種情況一一進行分析。
2.1 值類型參數的按值傳遞
對於參數又分為:形參和實參,形參指的是被調用方法中的參數,實參指的是調用方法的參數,下面結合代碼幫助大家理解形參和實參的概念:
class Program { static void Main(string[] args) { int addNum = 1; // addNum 就是實參, Add(addNum); } // addnum就是形參,也就是被調用方法中的參數 private static void Add(int addnum) { addnum = addnum + 1; Console.WriteLine(addnum); } }
對於值類型的按值傳遞,傳遞的是該值類型實例的一個拷貝,也就是形參此時接受到的是實參的一個副本,被調用方法操作是實參的一個拷貝,所以此時並不影響原來調用方法中的參數值,為了證明這點,看看下面的代碼和運行結果就明白了:
class Program { static void Main(string[] args) { // 1. 值類型按值傳遞情況 Console.WriteLine("按值傳遞的情況"); int addNum = 1; Add(addNum); Console.WriteLine(addNum); Console.Read(); } // 1. 值類型按值傳遞情況 private static void Add(int addnum) { addnum = addnum + 1; Console.WriteLine(addnum); }
運行結果為:
從結果中可以看出addNum調用方法之后它的值並沒有改變,Add 方法的調用只是改變了addNum的副本addnum的值,所以addnum的值修改為2了。然而我們的分析到這里並沒有結束,為了讓大家深入理解傳遞傳遞,我們有必要知道為什么值類型參數的按值傳遞不會修改實參的值,相信下面這張圖可以解釋你所有的疑惑:
2.2 引用類型參數的按值傳遞
當傳遞的參數是引用類型的時候,傳遞和操作的是指向對象的引用(看到這里,有些朋友會覺得此時不是傳遞引用嗎?怎么還是按值傳遞了?對於這個疑惑,此時確實是按值傳遞,此時傳遞的對象的地址,傳遞地址本身也是傳遞這個地址的值,所以此時仍然是按值傳遞的),此時方法的操作就會改變原來的對象。對於這點可能看文字描述會比較難理解下面結合代碼和分析圖來幫助大家理解下:
class Program { static void Main(string[] args) { // 2. 引用類型按值傳遞情況 RefClass refClass = new RefClass(); AddRef(refClass); Console.WriteLine(refClass.addnum); } // 2. 引用類型按值傳遞情況 private static void AddRef(RefClass addnumRef) { addnumRef.addnum += 1; Console.WriteLine(addnumRef.addnum); } } class RefClass { public int addnum=1; }
運行結果為:
為什么此時傳遞引用就會修改原來實參中的值呢?對於這點我們還是參數在內存中分布圖來解釋下:
2.3 .String引用類型的按值傳遞的特殊情況
對於String類型同樣是引用類型,然而對於string類型的按值傳遞時,此時引用類型的按值傳遞卻不會修改實參的值,可能很多朋友對於這點很困惑,下面具體看看下面的代碼:
class Program { static void Main(string[] args) { // 3. String引用類型的按值傳遞的特殊情況 string str = "old string"; ChangeStr(str); Console.WriteLine(str); } // 3. String引用類型的按值傳遞的特殊情況 private static void ChangeStr(string oldStr) { oldStr = "New string"; Console.WriteLine(oldStr); } }
運行結果為:
對於為什么原來的值沒有被改變主要是因為string的“不變性”,所以在被調用方法中執行 oldStr="New string"代碼時,此時並不會直接修改oldStr中的"old string"值為"New string",因為string類型是不變的,不可修改的,此時內存會重新分配一塊內存,然后把這塊內存中的值修改為 “New string”,然后把內存中地址賦值給oldStr變量,所以此時str仍然指向 "old string"字符,而oldStr卻改變了指向,它最后指向了 "New string"字符串。所以運行結果才會像上面這樣,下面內存分布圖可以幫助你更形象地理解文字表述:
三、按引用傳遞
不管是值類型還是引用類型,我們都可以使用ref 或out關鍵字來實現參數的按引用傳遞,然而按引用進行傳遞的時候,需要注意下面兩點:
方法的定義和方法調用都必須同時顯式使用ref或out,否則會出現編譯錯誤
CLR允許通過out 或ref參數來實現方法重載。如:
#region CLR 允許out或ref參數來實現方法重載 private static void Add(string str) { Console.WriteLine(str); } // 編譯器會認為下面的方法是另一個方法,從而實現方法重載 private static void Add(ref string str) { Console.WriteLine(str); } #endregion
按引用傳遞可以解決由於值傳遞時改變引用副本而不影響引用本身的問題,此時傳遞的是引用的引用(也就是地址的地址),而不是引用的拷貝(副本)。下面就具體看看按引用傳遞的代碼:
class Program { static void Main(string[] args) { #region 按引用傳遞 Console.WriteLine("按引用傳遞的情況"); int num = 1; string refStr = "Old string"; ChangeByValue(ref num); Console.WriteLine(num); changeByRef(ref refStr); Console.WriteLine(refStr); #endregion Console.Read(); } #region 按引用傳遞 // 1. 值類型的按引用傳遞情況 private static void ChangeByValue(ref int numValue) { numValue = 10; Console.WriteLine(numValue); } // 2. 引用類型的按引用傳遞情況 private static void changeByRef(ref string numRef) { numRef = "new string"; Console.WriteLine(numRef); } #endregion }
運行結果為:
從運行結果可以看出,此時引用本身的值也被改變了,通過下面一張圖來幫忙大家理解下按引用傳遞的方式:
四、總結
到這里參數的傳遞所有內容就介紹完了。總之,對於按值傳遞,不管是值類型還是引用類型的按值傳遞,都是傳遞實參的一個拷貝,只是值類型時,此時傳遞的是實參實例的一個拷貝(也就是值類型值的一個拷貝),而引用類型時,此時傳遞的實參引用的副本。對於按引用傳遞,傳遞的都是參數地址,也就是實例的指針。
所有源碼下載:參數傳遞