關於C#實值型別,參考型別,值傳遞,引用傳遞
說到參數傳遞,必須得弄清實值型別和參考型別:
(為了容易表達,我暫且命名存放在堆中的內容為堆(heap)中對象,存放在棧(stack)上的內容為棧中對象。)
實值型別存放在棧中,直接存取。如果有:int a=0;int b=a;就產生了兩個棧中對象。
參考型別需要在堆中顯式分配,且不能直接存取,需要在棧中分配一個棧中對象(C++叫指標,C#叫引用)指向其堆中對象。
如果:
StringBuilder strb = new StringBuilder();
StringBuilder strb2 = strb;
則在堆中只有一個堆中對象,只是棧中有兩個棧中對象指向堆中對象。
可以看出:每個變數都是一個棧中對象。不管是實值型別還是參考型別,只是實值型別的棧中對象就是其內容,而參考型別的棧中對象只是一個指向堆中對象的地址。
判斷是實值型別還是參考型別:
int a1 = 10;
StringBuilder strb1 = new StringBuilder("ABC");
int a2 = a1;
StringBuilder strb2 = strb1;
bool bl1 = object.ReferenceEquals(a1,a2); //false為實值型別(因為實值型別要裝箱)
bool bl2 = object.ReferenceEquals(strb1,strb2); //true為參考型別
參數傳遞分值傳遞和引用傳遞兩種。
通常,在沒有顯式指出ref和out時都是值傳遞。
值傳遞:傳的是對象的值拷貝。(即函數內參數對象是調用時傳遞對象的棧中對象的拷貝。)
引用傳遞:傳的是棧中對象的地址。(即函數內參數對象與調用時傳遞對象完全是同一棧中對象。)
現在用例子來說明傳值跟傳地址的不同:
private void button2_Click(object sender, System.EventArgs e)
{
StringBuilder strb1 = new StringBuilder();
StringBuilder strb2 = new StringBuilder();
Test1(strb1);
Test2(ref strb2);
string str1 = strb1.ToString(); //str1值:"A"
string str2 = strb2.ToString(); //str2值:"BC"
}
void Test1(StringBuilder strb)
{
//strb和strb1是兩個棧中對象,但指向相同的地址,這個操作是改變堆中對象
strb.Append("A");
//這裡將strb指向一個新的堆中對象,所以後面的操作與strb1指向的棧中對象無關
strb = new StringBuilder("B");
strb.Append("C");
}
void Test2(ref StringBuilder strb)
{
//這裡的strb和strb2是同一個棧中對象,所以改變strb的值使其指向另一個對象也等於改變strb2
strb = new StringBuilder("B");
strb.Append("C");
}