須知道.NET 內存分配

在分析內存分配時,應該先了解關於堆棧的區別多線程

堆的分配向高地址擴展,而棧的分配向低地址擴展。ide

二。內存分配this

關於內存的分配,首先應該瞭解分配在哪裏的問題。CLR管理內存的區域,主要有三塊,分別爲:spa

線程的堆棧,用於分配值類型實例。堆棧主要由操做系統管理,而不受垃圾收集器的控制,當值類型實例所在方法結束時,其存儲單位自動釋放。棧的執行效率高,但存儲容量有限。操作系統

GC堆,用於分配小對象實例。若是引用類型對象的實例大小小於85000字節,實例將被分配在GC堆上,當有內存分配或者回收時,垃圾收集器可能會對GC堆進行壓縮,詳情見後文講述。線程

  
  
  
  
  1. public class VIPUser:User  
  2.      {  
  3.         //分配1Byte  
  4.         public bool isVip;  
  5.         public bool IsVipUser()  
  6.         {  
  7.             return isVip;  
  8.         }  
  9.          static void Main(string[] args)  
  10.          {  
  11.              //分配內存空間和初始化操做  
  12.              VIPUser aUser;  
  13.              //將對象引用賦給aUser變量,創建aUser和VIPUser的關聯  
  14.              aUser = new VIPUser();  
  15.              //Q:類型的分配的字節數?  
  16.              //就本類而言須要15Byte。可是實例對象所佔的字節總數還要加上對象附加成員所需的字節數,其中包括附加成員TypeHandle和SyncBlockIndex共8個字節。在託管堆上分配的字節總數爲23字節,而堆上的內存塊老是按照4Byte的倍數進行分配,所以本類中將分配24字節的地址空間  
  17.    
  18.              //最後調用對象構造器,進行對象初始化操做,完成建立  
  19.    
  20.              //構造過程  
  21.              //a.構造VIPUser類型的Type對象,主要包括靜態字段、方法表、實現的接口等,並將其分配在上文提到託管堆的Loader Heap上。  
  22.    
  23.              //b.初始化aUser的兩個附加成員:TypeHandle和SyncBlockIndex。將TypeHandle指針指向Loader Heap上的MethodTable,CLR將根據TypeHandle來定位具體的Type;將SyncBlockIndex指針指向Synchronization Block的內存塊,用於在多線程環境下對實例對象的同步操做。  
  24.    
  25.              //c.調用VIPUser的構造器,進行實例字段的初始化。實例初始化時,會首先向上遞歸執行父類初始化,直到完成System.Object類型的初始化,而後再返回執行子類的初始化,直到執行VIPUser類爲止。以本例而言,初始化過程爲首先執行System.Object類,再執行User類,最後纔是VIPUser類。最終,newobj分配的託管堆的內存地址,被傳遞給VIPUser的this參數,並將其引用傳給棧上聲明的aUser。  
  26.    
  27.              aUser.isVip = true;  
  28.              Console.WriteLine(aUser.IsVipUser());  
  29.              //上述過程,基本完成了一個引用類型建立、內存分配和初始化的整個流程  
  30.          }  
  31.      }  
  32.      public class UserInfo  
  33.      {  
  34.          //分配4個字節  
  35.          private Int32 age = -1;  
  36.          //分配2個字節  
  37.          private char level = 'A';  
  38.      }  
  39.      public class User  
  40.      {  
  41.          //分配4byte  
  42.          private Int32 id;  
  43.          //保存了UserInfo的引用 佔用4Byte  
  44.          //僅是一個引用(指針),保存在線程的堆棧上,佔用4Byte的內存空間 用於保存user對象的有效地址 如今試圖對user的任何操做將拋出NullReferenceException  
  45.          private UserInfo user;  
  46.      } 

LOH(Large Object Heap)堆,用於分配大對象實例。若是引用類型對象的實例大小不小於85000字節時,該實例將被分配到LOH堆上,而LOH堆不會被壓縮,並且只在徹底GC回收時被回收。 指針

在瞭解內存分配以前  首先了解一下三個概念對象

TypeHandle,類型句柄,指向對應實例的方法表,每一個對象建立時都包含該附加成員,而且佔用4個字節的內存空間。咱們知道,每一個類型都對應於一個方法表,方法表建立於編譯時,主要包含了類型的特徵信息、實現的接口數目、方法表的slot數目等。繼承

SyncBlockIndex,用於線程同步,每一個對象建立時也包含該附加成員,它指向一塊被稱爲Synchronization Block的內存塊,用於管理對象同步,一樣佔用4個字節的內存空間。遞歸

NextObjPtr,由託管堆維護的一個指針,用於標識下一個新建對象分配時在託管堆中所處的位置。CLR初始化時,NextObjPtr位於託管堆的基地址。

三。繼承本質論

  
  
  
  
  1. //Bird bird建立的是一個對象的引用,而new Bird()是建立Bird對象,分配內存和初始化操做,而後將對象引用賦給bird變量,也就是簡歷bird和Bird 之間的關聯  
  2.              Bird bird = new Bird();  
  3.              //2.從繼承的角度來分析CLR在運行時如何執行對象的建立過程  
  4.                  //2.1 首先是字段的建立 字段的存儲順序由上到下排列,最高層類的字段排在最前面  
  5.                  //2.2方法表的建立是類第一次加載到AppDomain時完成的,在對象建立時只是將其附加成員TypeHandle指向方法列表Loader Heap上的地址,將對象與其動態方法列表相關聯起來,所以方法表示先於對象存在的。    
  6.              Chicken ch = new Chicken(); 
相關文章
相關標籤/搜索