筆記
由於在.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()並不會改變調用它的字串對象,而是返回含有新資料的新的字串對象。