日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學無先后,達者為師

網站首頁 編程語言 正文

.NET中堆棧和堆的特點與差異介紹_實用技巧

作者:.NET開發菜鳥 ? 更新時間: 2022-05-23 編程語言

一、前言

.NET提供了垃圾回收機制,使程序員從內存管理中被解放出來。但這并不代表程序員就無須了解分配的對象是如何被回收的。更重要的是,一些非托管的資源仍然需要程序員小心地分配與回收。

理解堆和堆棧是理解內存管理的基礎。每一個.NET程序都最終會運行在一個操作系統進程中,假設這個操作系統是傳統的32位的,那每個.NET程序都可以擁有一個4GB的虛擬內存。.NET會在這個4GB的內存塊中開辟出三塊內存分別作為堆棧、受托管的堆和非托管的堆。

二、.NET中的堆棧

.NET中的堆棧用來存儲值類型的對象和引用類型對象的引用,堆棧的分配是連續的,在.NET程序中,始終存儲了一個特殊的指針指向堆棧的尾部,這樣一個堆棧內存的分配就直接從這個指針指向的內存位置開始向下分配。下圖展示了.NET的堆棧分配方式。

如上圖所示,堆棧上的地址從高位開始往低位分配內存,.NET只需要保存一個堆棧指針指向下一個未分配內存的內存地址。對于所有需要分配的對象,依次分配到堆棧中,其釋放也嚴格按照棧的邏輯,依次進行退棧。這里提到的“依次”,是指按照變量的作用域進行的。考慮下面的代碼:

ClassA a = new ClassA();
a.intA = 1;
a.intB = 2;

這里假設ClassA是一個引用類型,則堆棧中依次需要分配的是a的引用、a.intA和a.intB。當a的作用域結束后,這三個變量則從堆棧中依次退出:a.intB、a.intA,然后才是a。

三、.NET中的托管堆

接下來我們看一下托管的堆。.NET中的引用類型對象是分配在托管堆上的。通常我們稱.NET中的堆,指的就是托管的堆。和堆棧一樣,托管的堆也是進程內存空間中的一塊區域。但托管堆中的內存的分配卻和堆棧有很大的區別。受益于.NET的內存管理機制,托管堆的分配也是連續的,但是堆中存在著暫時不能被分配卻已經無用的對象內存塊。當一個引用類型對象被初始化時,就會通過指向堆上可用空間的指針分配一塊連續的內存,然后使堆棧上的引用指向堆上的這塊內存塊。下圖展示了堆的分配方式。

如上圖所示,程序通過分配在堆棧中的引用來找到分配在托管堆的對象實例。當堆棧區域中的引用退出作用域時,就僅僅斷開引用和實際對象的聯系。而當托管堆中的內存不夠時,.NET開始執行垃圾回收。垃圾回收是一個非常復雜的過程,它不僅涉及托管堆中對象的釋放,而且需要移動合并托管堆中的內存塊。當垃圾回收后,堆內不被使用的對象才會被部分釋放,而在這之前,它們在堆內是暫時不可用的。

四、.NET中的非托管堆

.NET的程序還包含了非托管的堆,所有需要分配堆內存的非托管資源將會被分配到非托管堆上。非托管的堆需要程序員用指針手動地分配并且手動地釋放,.NET的垃圾回收和內存管理制度不適用于非托管的堆。

五、堆棧、托管堆和非托管堆的比較

堆棧、托管堆和非托管堆的分配各有特點。堆棧的內存是連續分配的,按照作用域依次分配和釋放。堆棧的機制非常簡單,.NET依靠一個堆棧指針就可以進行內存操作,分配一個對象和釋放一個對象的大部分操作就是自增或者自減堆棧指針。.NET中的值類型對象和引用類型對象的引用是分配在堆棧內的。

托管堆的內存分配雖然也是連續的,但它卻比堆棧復雜得多。一塊堆內存的分配需要涉及很多.NET內存管理機制的內部操作,另外當堆內存不夠時,垃圾回收的執行代價也是非常大的。相對于堆棧來說,堆的分配效率低得多。.NET中的引用類型對象是分配在托管堆上的,這些對象通過分配在堆棧上的引用來進行訪問。

非托管堆和托管堆的區別在于非托管堆不受.NET的管理。非托管堆的內存是由程序員手動分配和釋放的,垃圾回收機制不適用于非托管堆,內存塊也不會被合并移動,所以非托管堆的內存分配按塊的,不連續的。

六、總結

.NET程序在進程內存中分配出堆棧、托管堆和非托管堆。所有的值類型對象和引用類型對象的引用都分配在堆棧上,堆棧根據對象的生存周期來依次分配和釋放,堆棧根據一個指向棧尾的指針來分配內存,效率很高。

.NET所有的引用類型對象分配在托管堆上,托管堆連續分配內存,并且受.NET的垃圾收集機制管理,受托管堆的內存分配和釋放涉及復雜的內存管理,效率相對于堆棧來說低得多。

需要分配堆內存的非托管類型將被分配在非托管堆上,非托管堆不受.NET垃圾收集機制管理,內存塊完全由程序員手動申請和釋放。

原文鏈接:https://www.cnblogs.com/dotnet261010/p/12330112.html

欄目分類
最近更新