本文會闡述六個重要的概念:堆、棧、值類型、引用類型、裝箱和拆箱。本文首先會經過闡述當你定義一個變量以後系統內部發生的改變開始講解,而後將關注點轉移到存儲雙雄:堆和棧。以後,咱們會探討一下值類型和引用類型,並對有關於這兩種類型的重要基礎內容作一個講解。編程
本文會經過一個簡單的代碼來展現在裝箱和拆箱過程當中所帶來的性能上的影響,請各位仔細閱讀。函數
當你在一個.NET應用程序中定義一個變量時,在RAM中會爲其分配一些內存塊。這塊內存有三樣東西:變量的名稱、變量的數據類型以及變量的值。工具
上面簡單闡述了內存中發生的事情,可是你的變量究竟會被分配到哪一種類型的內存取決於數據類型。在.NET中有兩種可分配的內存:棧和堆。在接下來的幾個部分中,咱們會試着詳細地來理解這兩種類型的存儲。性能
爲了理解棧和堆,讓咱們經過如下的代碼來了解背後到底發生了什麼。測試
1
2
3
4
5
6
7
8
9
10
11
|
public
void
Method1()
{
// Line 1
int
i=
4
;
// Line 2
int
y=
2
;
//Line 3
class1 cls1 =
new
class1();
}
|
代碼只有三行,如今咱們能夠一行一行地來了解到底內部是怎麼來執行的。spa
如今咱們許多的開發者朋友必定很好奇爲何會有兩種不一樣類型的存儲?咱們爲何不能將全部的內存塊分配只到一種類型的存儲上?pwa
若是你觀察足夠仔細,基元數據類型並不複雜,他們僅僅保存像 ‘int i = 0
’這樣的值。對象數據類型就複雜了,他們引用其餘對象或其餘基元數據類型。換句話說,他們保存其餘多個值的引用而且這些值必須一一地存儲在內存中。對象類型須要的是動態內存而基元類型須要靜態內存。若是需求是動態內存的話,那麼它將會在堆上爲其分配內存,相反,則會在棧上爲其分配。3d
既然咱們已經瞭解了棧和堆的概念了,是時候瞭解值類型和引用類型的概念了。值類型將數據和內存都保存在同一位置,而一個引用類型則會有一個指向實際內存區域的指針。指針
經過下圖,咱們能夠看到一個名爲i的整形數據類型,它的值被賦值到另外一個名爲j的整形數據類型。他們的值都被存儲到了棧上。code
當咱們將一個int類型的值賦值到另外一個int類型的值時,它其實是建立了一個徹底不一樣的副本。換句話說,若是你改變了其中某一個的值,另外一個不會發生改變。因而,這些種類的數據類型被稱爲「值類型」。
當咱們建立一個對象而且將此對象賦值給另一個對象時,他們彼此都指向了以下圖代碼段所示的內存中同一塊區域。所以,當咱們將obj賦值給obj1時,他們都指向了堆中的同一塊區域。換句話說,若是此時咱們改變了其中任何一個,另外一個都會受到影響,這也說明了他們爲什麼被稱爲「引用類型」。
在.NET中,變量是存儲到棧仍是堆中徹底取決於其所屬的數據類型。好比:‘String’或‘Object’屬於引用類型,而其餘.NET基元數據類型則會被分配到棧上。下圖則詳細地展現了在.NET預置類型中,哪些是值類型,哪些又是引用類型。
如今,你已經有了很多的理論基礎了。如今,是時候瞭解上面的知識在實際編程中的使用了。在應用中最大的一個意義就在於:理解數據從棧移動到堆的過程當中所發生的性能消耗問題,反之亦然。
考慮一下如下的代碼片斷,當咱們將一個值類型轉換爲引用類型,數據將會從棧移動到堆中。相反,當咱們將一個引用類型轉換爲值類型時,數據也會從堆移動到棧中。
不論是在從棧移動到堆仍是從堆中移動到棧上都會不可避免地對系統性能產生一些影響。
因而,兩個新名詞橫空出世:當數據從值類型轉換爲引用類型的過程被稱爲「裝箱」,而從引用類型轉換爲值類型的過程則被成爲「拆箱」。
若是你編譯一下上面這段代碼而且在ILDASM(一個IL的反編譯工具)中對其進行查看,你會發如今IL代碼中,裝箱和拆箱是什麼樣子的。下圖則展現了示例代碼被編譯後所產生的IL代碼。
爲了弄明白到底裝箱和拆箱會帶來怎樣的性能影響,咱們分別循環運行10000次下圖所示的兩個函數方法。其中第一個方法中有裝箱操做,另外一個則沒有。咱們使用一個Stopwatch對象來監視時間的消耗。
具備裝箱操做的方法花費了3542毫秒來執行完成,而沒有裝箱操做的方法只花費了2477毫秒,整整相差了1秒多。並且,這個值也會由於循環次數的增長而增長。也就是說,咱們要儘可能避免裝箱和拆箱操做。在一個項目中,若是你須要裝箱和裝箱,請仔細考慮它是不是絕對必不可少的操做,若是不是,那麼儘可能不用。
雖然以上代碼段沒有展現拆箱操做,但其效果一樣適用於拆箱。你能夠經過寫代碼來實現拆箱,而且經過Stopwatch來測試其時間消耗。
原文:http://blog.jobbole.com/77946/