做爲一個 Java 程序員,不瞭解 Java 內存模型就不能寫出可以充分利用內存的代碼。本文經過對 Java 內存模型的介紹,讓讀者可以瞭解 Java 的內存的分配狀況,適合 Java 初學者或者對 JMM 不熟悉的同窗。後面的博客會針對每一個部分作更加深刻的解釋。程序員
首先經過下圖對於 Java 內存模型有一個總體的認識,而後針對不一樣的區域的做用和存儲的內容作進一步的解釋。算法
這裏的 PC 不是 Personal Computer,而是 Program Counter Register,從名字就能夠看出來,這是一個寄存器,用來存儲須要執行的指令地址。編程
程序計數器(Program Counter (PC))是在電腦處理器中的一個寄存器,用來指示電腦下一步要運行的指令序列。--WikiPedia
PC 和其餘 JVM 內存區域最大的區別是:數組
「此內存區域是惟一一個在Java虛擬機規範中沒有規定任何OutOfMemoryError狀況的區域。」
摘錄來自: 周志明. 「深刻理解Java虛擬機:JVM高級特性與最佳實踐(第2版)」。 iBooks.
像上面的圖片同樣,PC 是每一個線程私有的,對於 Java 方法而言,PC 中存儲的是正在執行的虛擬機字節碼的內存地址;對於 Native 方法來講,PC 中的值爲空(Undefined)。學習
不管是在大學的 Java 編程課堂上,仍是咱們在學習過程編碼過程,常常會出現 StackOverFlow,甚至目前最大的技術問答社區的名字也是 StackOverFlow。Java 語言中會產生棧溢出的就是這塊內存區域,當你的程序中設置了超過 JVM 規定的遞歸深度的時候就會觸發這個異常。相似 JMM 的其餘內存區域,若是虛擬機棧在動態擴展的時候沒法申請到足夠的內存也會報OOM異常。編碼
Java 語言中每個方法的執行都對應着一個棧幀(Stack Frame)的建立,棧幀中存儲的是局部變量、方法出口等信息,所以對於一個方法的執行而言,所可以使用到的內存是在編譯期間就可以徹底肯定的,在運行期間不會發生變化。在棧幀中,局部變量空間成爲 Slot,除了 double 和 long 佔有 2 個 slot 外,其餘基本數據類型和對象引用都佔用 1 個 slot 空間spa
本地方法棧和虛擬機棧最大的區別就是虛擬機棧是爲執行 Java 字節碼服務的,而本地方法棧是爲了虛擬機使用到的 Native 方法服務的。除此以外,Java 虛擬機規範並無針對本地方法棧的實現作具體規定。在 HotSpot 虛擬機中,本地方法棧和虛擬機棧是共用同一塊內存的,不作具體區分。一樣,本地方法棧也會產生 OOM 異常和 StackOverFlow 異常。線程
「The heap is the runtime data area from which memory for all class instances and arrays is allocated。」 --Java虛擬機規範
Java 虛擬機規範規定全部的實例對象和數組都應該分配到 Java 堆中。
說的通俗一點就是全部 new 出來的對象和數組都會放到該區域,因爲如今的收集器都採用分代收集算法,因此在 Java 堆中又分了新生代和老年代,新生代有作了詳細的區分。該區域的大小能夠經過 JVM 參數 -Xmx
和 -Xms
來設置。code
在 JDK1.4 中引入了 NIO,能夠經過 Native 方法直接在堆外分配內存,而後經過在堆中存儲的引用來對這塊內存區域作操做。注意 這塊區域並不會在 -Xmx
和 -Xms
設置的大小以內,所以在設置 JVM 參數的時候要注意考慮這塊內存區域,避免設置的內存區域總額大於物理內存對象
Method Area 又叫 NonHeap,也是線程共有的內存區域,用來存:
在 JDK1.7 中已經將字符串常量池移出永久代,在 Java8 中更是以內取消了永久代,而是使用了元空間(MetaSpace)來存儲這些信息,從而永久代的大小不須要再製定,只要不超出物理內存的限制就不會產生 OOM 異常
運行時常量池主要用來存儲累的版本、字段、方法、接口等描述信息。常量池(Constant Pool Table)用來存儲各類字面量和符號引用。String 的 intern()
方法就是在運行期間將對象放到常量池中的。此部分也會出現 OOM 異常。