原則:盡可能控制對數據的修改,如果可以預測某個數據不會或不應該被改變,就要對其控制,而不要期望使用這個數據的調用者不會改變其值。
如果參數在使用過程中被意外修改,將會帶來不可預知的結果,而且這種錯誤很難被檢查到,所以我們在設計方法參數的時候,要充分考慮傳遞引用類型參數或者引用方式傳遞引用類型參數可能帶來的後果。
如果一個數據在傳遞過程中不能被改變,就要在構建這個對象的時候就使其值(字段或屬性)不被改變。
一、對於簡單的參數的控制
1、值類型參數傳遞
這種情況因為傳遞的是參數的副本,不影響原始值,不需要控制。
2、引用類型參數傳遞
a、由值類型組成的數據結構
需要將字段設置為只讀,屬性只有get。賦值只能通過構造方法進行。
b、包含引用類型字段的數據結構
這種情況是遞歸的,需要保證字段為readonly,屬性為get的同時,引用類型字段所使用類型也滿足該要求。
? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30public
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)引用類型參數的區別 [這一點最容易混淆]
區別在於使用該參數過程中為該引用新建了對象的情況下,前者不影響原始值,後者影響原始值,示例:
? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16void
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)影響原始值。
以上內容是小編給大家介紹的C#中的數組作為參數傳遞所引發的問題 ,希望對大家有所幫助!