值型別,例如:
int、
float
、bool
之類的基礎型別,以及用
struct
定義的型別,如:
datetime
。除此外,如
string
,陣列,以及用
class
定義的型別等都是引用型別。對於
c#來說,很難羅列出所有型別進行一一分別,這需要自己在編碼過程中進行分析總結。
值型別
引用型別
記憶體分配地點
分配在棧中
分配在堆中
效率
效率高,不需要位址轉換
效率低,需要進行位址轉換
記憶體**
使用完後,立即**
使用完後,不是立即**,等待gc**
賦值操作
進行複製,建立乙個同值新物件
只是對原有物件的引用
函式引數與返回值
是物件的複製
是原有物件的引用,並不產生新的物件
型別擴充套件
不易擴充套件
容易擴充套件,方便與型別擴充套件
通過如上細緻對比,大家對於值型別和引用型別有個清楚的概念。
不過,無論是對於值型別還是引用型別來說,對於其作為函式引數或者返回值的時候,都是容易犯錯誤的地方。
對於值型別來說,當其作為函式引數的時候,希望在函式中被修改,那麼直接如下操作是不能被修改的。
public
void increment( int i )
要想在函式中對傳進去的引數做真正的修改,需要借助於ref這個關鍵字,那麼正確的形式如下。
public
void increment( ref
int i )
也就是說,如果需要在函式中對值型別引數進行修改,需要用
ref或者
out進行標識才能真正實現。
而對於引用型別來說,當其作為函式引數的時候,它所遇到的情況恰恰與值型別相反,即不希望在函式中被修改,舉例如下。
public
void addvalue( mytype typvalue )
由於對於引用型別物件來說,其的賦值操作只是對原有物件的引用,因此在函式對其修改,實際上是直接修改了原有物件資料,這是很多情況不希望發生的(這裡例如對陣列或者
datatable
操作這類)。
為了防止這種事發生,需要給此型別提供
clone
函式
。例如對於如上的型別,可以入下實現。
public
class mytype:icloneable
get }
public mytype()
public mytype( int value)
#region
icloneable members
public
object clone()
#endregion }
那麼在呼叫的時候,用當前的物件的clone作為引數即可。
不過對於引用型別來說,提供乙個
clone
函式不是一件容易的事情,尤其出現引用型別巢狀的時候,所以說去實現乙個完全
clone
功能是件很費事又不討好的活,這也就是在論壇中常說的深
copy
和淺
copy
的問題。話雖如此,如果對於前面所說的有個大概了解,相信實現也不是不可能。
在c#中,尤其自己定義型別的時候,常常由於是用struct來定義還是用class來定義,即是定義乙個值型別還是乙個引用型別呢。在這本書上給了幾個判定條件,如果如下幾點都滿足的話,建議用struct來定義為值型別,否則用class定義為引用型別。
1. 這個型別是否主要為了資料儲存;
2. 是否只通過屬性來訪問物件的資料成員;
3. 這個型別是否不會有子型別;
4. 在程式處理的時候不會把這個型別物件通過多型來處理。
引用型別和值型別
c 是一種型別安全的語言。每乙個變數都要求定義為乙個特定的型別,並且要求儲存在變數中的值只能是這種型別的值。變數既能儲存值型別,也可以儲存引用型別,還可以是指標。這一課將講述前兩種型別,關於指標的討論我們將在下一課中進行。下面是關於值型別和引用型別不同點的概論 如果乙個變數v儲存的是值型別,則它直接...
引用型別和值型別
c 中值型別和引用型別作為方法引數傳遞的時候其實都可以說是 值 的傳遞,只不過這裡的 值 指代的東西有所區別。當方法的引數為值型別時,方法傳遞的是值本身的值。當方法的引數為引用型別時,方法傳遞的則是應用型別的引用的位址,也就是引用型別位址在棧上的值。舉個引用型別作為引數傳遞的例子 static vo...
值型別和引用型別
為了更好地說明兩種型別之間的區別,借用如下的 來說明 值型別引用型別 記憶體分配地點 分配在棧中 分配在堆中 效率效率高,不需要位址轉換 效率低,需要進行位址轉換 記憶體 使用完後,立即 使用完後,不是立即 等待gc 賦值操作 進行複製,建立乙個同值新物件 只是對原有物件的引用 函式引數與返回值 是...