C#中,資料傳遞的方式有兩種:按值傳遞和按引用傳遞,且預設情況下都是按值傳遞。而CLR支援兩種基本類型為:實值型別和參考型別。
1. 當傳遞的參數為實值型別,且資料傳遞方式為按值傳遞時,傳遞的是資料的一個拷貝。
2. 當資料傳遞方式為按引用傳遞時,必須以ref或者out關鍵字來修飾。由於這部分不是本文主題,不做詳細介紹,想瞭解相關內容可參見【原創】C#中ref和out的異同
3. 當參考型別參數按值傳遞時,傳遞的是參考型別地址的資料拷貝。
下面通過例子來驗證第3點中所說
先看以下範例程式碼:
class Program
{
public static void Main()
{
var abf = new ArgsByRef();
var abf1 = abf;
AddRef(abf);
Console.WriteLine(abf.i);
ReplaceRef(abf1);
Console.WriteLine(abf.i);
Console.WriteLine(abf1.i);
}
private static void AddRef(ArgsByRef abf)
{
abf.i = 20;
Console.WriteLine(abf.i);
}
private static void ReplaceRef(ArgsByRef abf)
{
if (abf == null) throw new ArgumentNullException("abf");
var m = new ArgsByRef();
abf = m;
Console.WriteLine(abf.i);
}
}
public class ArgsByRef
{
public int i = 10;
}
現在開始來詳細分析以上代碼:
1. var abf = new ArgsByRef(); 語句建立一個ArgsByRef執行個體abf,假設abf執行個體指向的堆中地址為0x1323。
2. var abf1 = abf; 語句建立一個ArgsByRef執行個體abf1,並將abf1執行個體的引用地址指向abf執行個體的引用地址,即abf1執行個體同樣指向地址0x1323。
3.
AddRef(abf);
private static void AddRef(ArgsByRef abf)
{
abf.i = 20;
Console.WriteLine(abf.i);
}
執行AddRef方法時,傳遞abf執行個體作為參數。由於傳遞的是對象的引用,所以在方法內對對象的修改會直接影響對象。
4. 當執行ReplaceRef方法時,傳遞abf1執行個體作為參數,建立ArgsByRef執行個體m,假設執行個體m指向的堆地址為0x1345。
5. 當執行abf=m時,其實是建立了一個abf1執行個體引用地址的拷貝,使當前abf1執行個體的引用地址為0x1345。這裡重點在於引用地址的拷貝,我畫的表示圖如下(不知道對與不對,還望大哥們多指點一下):
6. 退出ReplaceRef方法,回到主方法後,由於是參考型別的引用地址的值傳遞,所以並沒有改變對象的引用地址。執行個體abf1的引用地址仍為0x1323。
所以abf1.i仍為20
最後:說了這麼多,也不知道描述清楚沒有,如果有錯誤,還希望各位看官能不吝指點。我認為重點理解引用地址的拷貝。希望對大家理解實值型別和參考型別,以及按值傳遞和按引用傳遞有協助。