原則:儘可能控制對資料的修改,如果可以預測某個資料不會或不應該被改變,就要對其控制,而不要期望使用這個資料的調用者不會改變其值。
如果參數在使用過程中被意外修改,將會帶來不可預知的結果,而且這種錯誤很難被檢查到,所以我們在設計方法參數的時候,要充分考慮傳遞參考型別參數或者引用方式傳遞參考型別參數可能帶來的後果。
如果一個資料在傳遞過程中不能被改變,就要在構建這個對象的時候就使其值(欄位或屬性)不被改變。
一、對於簡單的參數的控制
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<T>介面代替List<T>。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)影響原始值。