C#中值型別和引用型別引數傳遞
原則:儘可能控制對資料的修改,如果可以預測某個資料不會或不應該被改變,就要對其控制,而不要期望使用這個資料的呼叫者不會改變其值。
如果引數在使用過程中被意外修改,將會帶來不可預知的結果,而且這種錯誤很難被檢查到,所以我們在設計方法引數的時候,要充分考慮傳遞引用型別引數或者引用方式傳遞引用型別引數可能帶來的後果。
如果一個數據在傳遞過程中不能被改變,就要在構建這個物件的時候就使其值(欄位或屬性)不被改變。
一、對於簡單的引數的控制
1、值型別引數傳遞
這種情況因為傳遞的是引數的副本,不影響原始值,不需要控制。
2、引用型別引數傳遞
a、由值型別組成的資料結構
需要將欄位設定為只讀,屬性只有get。賦值只能通過構造方法進行。
b、包含引用型別欄位的資料結構
這種情況是遞迴的,需要保證欄位為readonly,屬性為get的同時,引用型別欄位所使用型別也滿足該要求。
public class SuperClass
{
private readonly int _no;
private readonly SubClass _tag;
public int NO
{
get{ return _no;}
}
public SubClass Tag
{
get{ retirn _tag;}
}
public SuperClass(int no,SubClass tag)
{
_no=no;
_tag=tag;
}
}
public class SubClass
{
private readonly int _field;
public int Field
{
get{ return _field;}
}
public SubClass(int field)
{
_field=field;
}
}
二、對於複雜引用型別引數傳遞的控制
所謂複雜,是引數是陣列或集合型別,或者引數包含這些型別資料,這種情況下上面的方法不能保證引數資料不被修改,因為即使物件為只讀的,但是物件中的陣列或集合欄位(屬性)還是可以修改的。
1、集合引數(包含集合欄位的引用引數也一樣)
.net 4.5以前版本可以使用不包含修改集合元素方法的介面來代替具體集合型別。例如使用IEnumerable介面代替List。4.5版本可以直接使用IReadOnlyCollection介面或實現該介面的集合型別。
2、陣列引數
沒有好的辦法保護陣列型別引數不被修改,所以儘量避免使用陣列型別作為方法引數,除非用到可選引數時候。
三、理解上面的東西需要區分清楚一下概念的區別
1、值型別和引用型別的區別
2、值傳遞和引用傳遞(ref和out)的區別
3、傳遞引用型別引數和引用傳遞(ref和out)引用型別引數的區別 [這一點最容易混淆]
區別在於使用該引數過程中為該引用新建了物件的情況下,前者不影響原始值,後者影響原始值,示例:
void FunA(MyClass a)
{
a=new MyClass(“A”);
}
void FunB(ref MyClass a)
{
a=new MyClass(“B”);
}
void Test()
{
MyClass a=new MyClass(“A”);
FunA(a);
Print(a); //a還是原始的物件 TEST
FunB(ref a);
Print(a); //a變為新物件 B
}
記住一條原則:
值型別傳遞的是值的副本,引用型別傳遞的是物件引用,所以值引數的修改不影響原始值,引用型別的修改影響原始值;值傳遞的引數構建不影響原始值,引用傳遞(ref和out)影響原始值。