由於在.Net中存在兩種類型,分別是值類型(value type)和引用類型(reference type),所以很多關於C#中參數傳遞的混淆就因此而生。
首先要弄清楚的是:值類型是分配在棧(stack)上面,而引用類型分配在堆(heap)上面。棧是一種先進後出,並且由系統自動操作的存儲空間。而堆(在.Net上准確的說是托管堆 Managed Heap)是一種自由儲存區(Free Memory),在該區域中,必須明確的為對象申請存儲空間(一般在Java和C#中都是使用的new關鍵字),並可以在使用完以後釋放申請的存儲空間(Java和C#都使用垃圾回收機制 Garbage Collector自動釋放對象空間)
引用類型(reference type):它存放的值是指向數據的引用(reference),而不是數據本身。示例:
System.Text.StringBuilder sb = new StringBuilder();
這裡,我們聲明一個變量sb,並通過new StringBuilder()創建了一個StringBuilder(與Java中StringBuffer類似)對象,再將對象的引用(reference)賦值給變量sb,即變量sb中保存的是StringBuilder對象的引用,而非對象本身。
System.Text.StringBuilder first = new StringBuilder();
System.Text.StringBuilder second = first;
這裡,我們將變量first的值(對一個StringBuilder對象的引用)賦值給變量second,即first和second都指向同一個StringBuilder對象。對StringBuilder對象的任何修改都會影響到first和second變量。
System.Text.StringBuilder first = new StringBuilder();
System.Text.StringBuilder second = first;
first.Append("hello");
first = null;
Console.WriteLine(second);
這裡,輸出的結果是 hello。由於first和second都含有對同一StringBuilder對象的引用。然後通過first的引用調用StringBuilder對象的Append方法,將對象進行修改,即添加字符串“hello”,然後又將first賦值為null,表示讓first不引用任何對象。最後通過second的引用隱式調用StringBuilder對象的ToString方法輸出“hello”。由此可見,first的值改變了(被賦值為null),而它所引用的對象並不會發生改變,second照樣引用到StringBuilder對象。
class類型,interface類型,delegate類型和array類型都是引用類型。
值類型(value type):引用類型中變量和實際數據之間還隔了一間接層,而值類型就完全不存在,值類型的變量直接保存的就是數據。
struct IntHolder
{
public int i;
}
這裡,結構是值類型,IntHolder是一個結構:
IntHolder first = new IntHolder();
first.i = 5;
IntHolder second = first;
first.i = 6;
Console.WriteLine(second.i);
輸出結果為5。這裡second = first 以後second保存的是first的值拷貝,即second.i = 5;而後來的first.i發生了改變並不會影響second.i。所以輸出值為5。
簡單類型(比如int,double,char),enum類型,struct類型都是值類型。
注意:有一些類型(比如string類型)的行為看起來像值類型,但實際上是引用類型。這些類型被稱為immutable類型,也就是說這種類型的實例只要被構造好就不會改變。比如,string.Replace()並不會改變調用它的字符串對象,而是返回含有新數據的新的字符串對象。