




版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領
文檔簡介
第C#中值類型和引用類型的區別一、值類型和引用類型的區別
.NET的類型可以分為兩類:值類型和引用類型。這兩種類型各有特點,即使它們都繼承自System.Object,并且有裝箱和拆箱等操作確保兩種類型可以方便地交互,但是理解值類型和引用類型將有助于程序員編寫出高效的代碼,相反的,在不理解值類型和引用類型的情況下,程序員很容易編寫出可以正確執行但性能較差的代碼。
所有.NET的類型都可以分為兩類:值類型和引用類型。最簡單也最明確的一個區分標準是:所有的值類型都繼承自System.ValueType(System.ValueType繼承自System.Object),也就是說,所有繼承自System.ValueType的類型都是值類型,而其他類型都是引用類型。常用的值類型包括結構、枚舉、整數型、浮點型、布爾型等,而在C#中所有以class關鍵字定義的類型都是引用類型。
1、賦值時的區別
引用類型和值類型最顯著的一個區別在于變量的賦值問題。值類型的變量將直接獲得一個真實的數據副本,而對引用類型的賦值僅僅是把對象的引用賦給變量,這樣就可能導致多個變量引用到一個實際對象實例上。
來看下面一個簡單的示例:首先為了測試建立一個簡單的引用類型和一個簡單的值類型。然后在Main方法中,測試對值類型和引用類型對象進行賦值的不同結果,代碼如下:
usingSystem;
namespaceConsoleApp1
///summary
///一個簡單的引用類型
////summary
publicclassRef
publicintiValue{get;set;}
publicRef(inti)
iValue=i;
publicoverridestringToString()
return$"iValue的值為:{iValue.ToString()}";
///summary
///一個簡單的值類型
////summary
publicstructVal
publicintValue{get;set;}
publicVal(inti)
Value=i;
publicoverridestringToString()
return$"Value的值為:{Value.ToString()}";
classProgram
staticvoidMain(string[]args)
//測試引用類型的賦值
Refref1=newRef(1);
Refref2=ref1;
//賦值
ref2.iValue=2;
//測試值類型的賦值
Valval1=newVal(1);
Valval2=val1;
val2.Value=2;
//輸出
Console.WriteLine($"ref1:{ref1}");
Console.WriteLine($"ref2:{ref2}");
Console.WriteLine($"val1:{val1}");
Console.WriteLine($"val2:{val2}");
Console.ReadKey();
}
簡單分析上面的代碼,程序定義了一個引用類型Ref和一個值類型Val,兩者的內容幾乎完全相同。在Main方法中,分別測試了引用類型和值類型的賦值。當代碼把一個引用類型變量賦值給另一個引用變量:Refref2=ref1時,實際上是把ref1的對象引用賦給了ref2,這樣,兩個引用變量實際指向了同一個對象。如圖所示:
而值類型的賦值則不同,val1和val2都保留了屬于自己的數據副本,所以當val2改變時,val1不受到影響。如圖所示:
上面代碼的輸出結果:
2、內存分配的區別
除了賦值的區別,引用類型和值類型在內存的分配位置上也有區別。引用類型的對象將會在堆上分配內存,而值類型的對象則會在堆棧上分配內存。堆棧的空間相對有限,但運行效率卻比高的多。
3、來自繼承結構的區別
最后,由于所有的值類型都有一個共同的基類:System.ValueType,所以值類型擁有一些引用類型不具有的共同性質,較重要的一點是值類型的比較方法:Equals方法的實現有了改變。所有的值類型都實現了內容的比較,而引用類型在沒有重寫Equals方法的情況下,仍然采用引用比較。還是以上面的代碼為了,看下面的代碼:
usingSystem;
namespaceConsoleApp1
///summary
///一個簡單的引用類型
////summary
publicclassRef
publicintiValue{get;set;}
publicRef(inti)
iValue=i;
publicoverridestringToString()
return$"iValue的值為:{iValue.ToString()}";
///summary
///一個簡單的值類型
////summary
publicstructVal
publicintValue{get;set;}
publicVal(inti)
Value=i;
publicoverridestringToString()
return$"Value的值為:{Value.ToString()}";
classProgram
staticvoidMain(string[]args)
////測試引用類型的賦值
//Refref1=newRef(1);
//Refref2=ref1;
////賦值
//ref2.iValue=2;
////測試值類型的賦值
//Valval1=newVal(1);
//Valval2=val1;
//val2.Value=2;
//輸出
//Console.WriteLine($"ref1:{ref1}");
//Console.WriteLine($"ref2:{ref2}");
//Console.WriteLine($"val1:{val1}");
//Console.WriteLine($"val2:{val2}");
//測試引用類型的賦值
Refref1=newRef(1);
Refref2=newRef(1);
//測試值類型的賦值
Valval1=newVal(1);
Valval2=newVal(1);
Console.WriteLine(ref1.Equals(ref2));
Console.WriteLine(val1.Equals(val2));
Console.ReadKey();
}
程序輸出結果:
在Main方法中,分別定義了一對內容完全相同的值類型對象和引用類型對象,調用Equals方法來比較,發現值類型對象比較返回true,而引用類型對象比較返回false。
所有繼承自System.ValueType的類型都是值類型,而其他類型都是引用類型。值類型的賦值會產生一個新的數據副本,所以每個值類型都擁有一個數據副本。而引用類型的賦值
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯系上傳者。文件的所有權益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經權益所有人同意不得將文件中的內容挪作商業或盈利用途。
- 5. 人人文庫網僅提供信息存儲空間,僅對用戶上傳內容的表現方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
- 6. 下載文件中如有侵權或不適當內容,請與我們聯系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
評論
0/150
提交評論