JVM內存模型與運行時數據區域

1、java內存模型

jvm-1

  1. java定義內存模型的目的是:爲了屏蔽各類硬件和操做系統的內存訪問之間的差別。
  2. java內存模型規定了全部的變量都存儲在主內存中,每條線程擁有本身的工做內存,工做內存保存了主內存中變量的副本。
  3. 線程對變量操做只能在工做內存中進行,不能直接讀寫主內存的變量。
  4. 不一樣線程之間的變量訪問須要經過主內存來完成。

一、java內存模型和java運行時數據區域的關係:主內存對應着java堆,工做內存對應着java棧。java

二、volatile關鍵字,使得變量的更新在各個工做內存中都是實時可見的。在DCL的單例模式中有運用到!面試


2、java運行時數據區域/內存區域

由於jvm的運行時數據區域一直在改善,因此不一樣jdk版本之間會有不一樣。算法

一、jdk1.7以前的jvm內存區域,擁有永久代

jvm-2

一、程序計數器的做用,由於.java文件被編譯成.class文件,它做爲當前線程所執行的字節碼的行號指示器。當字節碼解釋器工做時,就是經過改變這個計算器的值來選取下一條要執行的字節碼指令。每條線程都有一個獨立的程序計數器。數組

二、本地方法棧就是執行本地native方法的棧,native方法由虛擬機實現!jvm

三、java虛擬機棧描述的是該線程執行java方法(method)時的內存模型。每個方法都對應一個棧幀,棧幀中的局部變量表存儲了方法中的基本數據類型變量、對象引用變量。函數

jvm-3

如上圖所示,局部變量表保存了方法中聲明的8種基本類型變量和對象引用變量。每個棧幀中還有一個指向運行時常量池的引用,這是指String類型。 下面有一個經典的String對象生成的面試題!

四、java堆是JVM中內存最大的一塊,被全部線程共享。幾乎全部的對象實例都在這裏分配,因此java堆也是JVM垃圾回收的主要區域。java堆又被分紅了年輕代,老年代;年輕代進一步能夠劃分爲Eden空間,From Survivor空間、To Survivor空間。性能

jvm-4

當咱們使用new關鍵字分配對象時,就是在java堆中生成對象。

下面分析一下對象生成時的狀況。測試

  1. 由於Eden最大,因此新生成的對象都分配到Eden空間,當Eden空間快滿時,進行一次Minor GC,而後將存活的對象複製到From Survivor空間。這時,Eden空間繼續向外提供堆內存。
  2. 後面繼續生成的對象仍是放到Eden空間,當Eden空間又要滿的時候,這時候Eden空間和From Survivor空間同時進行一次Minor GC,而後把存活對象放到To Survivor空間。這時,Eden空間繼續向外提供堆內存。
  3. 接下來的狀況和2一致。Eden空間快滿的時候,Eden空間和To Survivor空間進行一次Minor GC,而後存活的對象放到From Survivor空間。
  4. 接下來的狀況和3一致。Eden空間快慢的時候,Eden空間和From Survivor空間進行一次Minor GC,而後存活的對象放到To Survivor空間。
  5. 就是說2個Survivor中的一個用來提供對象保存。當Eden空間和某一塊Survivor空間GC後,另外一塊Survivor空間放不下GC後存活的對象;或者是連續Minor GC15次左右的狀況;就把這部分存活對象放入到老年代空間。
  6. 當老年代空間也放滿的時候,進行Major GC,對老年代空間進行回收。(也叫作Full GC,Full GC的內存消耗很大,應該避免)

年輕代使用的是複製算法:每次Minor GC把Eden區和一塊Survivor區的存活對象複製到另外一塊Survivor區。老年代使用的是標記-整理算法:每次Major GC把存活對象都想內存空間的一端移動,而後直接清理掉端邊界之外的內存。spa

大對象如數組、很長的字符串,直接進入老年代空間。

五、方法區用於存儲JVM加載的類信息、final常量、static靜態變量等數據,方法區中的數據都是整個程序中惟一的。方法區還包含了運行時常量池,主要存放編譯期生成的字面量和符號引用(在類加載後放入)。String對象的字面量就會被放入到運行時常量池中。操作系統

垃圾回收在方法區主要是對常量的回收和對類型的卸載。

二、jdk1.8及以後的jvm內存區域,元空間取代了永久代

jvm-5

參考文章:java8的jvm內存區域

元空間和永久代的性質是同樣的,都是對JVM方法區的實現,做用是同樣的。不過元空間與永久代之間最大的區別在於:元空間並不在虛擬機JVM內存中,而是使用本地內存。

爲何用元空間取代永久代呢?

  1. 字符串存在永久代中,容易出現性能問題和內存溢出。
  2. 類及方法的信息等比較難肯定其大小,所以對於永久代的大小指定比較困難,過小容易出現永久代溢出,太大則容易致使老年代溢出。
  3. 永久代會爲GC帶來沒必要要的複雜度,而且回收效率偏低。

直接內存

JDK1.4以後加入的NIO,引入了基於通道channel和緩衝區buffer的IO,直接使用native函數分配堆外內存,顯著提升IO性能,避免了原來BIO的在java堆和naive堆中來回複製數據。

jvm-6

三、字符串String生成時的內存分配狀況

參考文章:字符串常量池

四、生成對象時的內存狀況

下面來分析一下咱們常見的生成對象或基本數據類型變量的內存模型。這樣能夠對JVM有一個更好的理解。

  1. int i =3;,一個方法對應一個棧幀,方法中的基本數據類型變量直接在棧幀中分配。若是是static、final類型的基本數據類型則存儲在運行時常量池中,和String同樣
  2. Object o1 = new Object();,對象引用(Object o1)存儲在棧幀中,可是對象數據(new Object())存儲在java堆中,對象類型數據(Class等信息)存儲在方法區中。
  3. String s1 = new String("abcd");,使用new聲明的對象,對象引用(String s1)存儲在棧幀中,對象數據(new String(「abcd」))存儲在java堆中,字符串值(「abcd」)存儲在運行時常量池中。
  4. String s2 = 「abc」,對象引用(String s2)存儲在棧幀中,字符串值(「abc」)存儲在運行時常量池中。

jvm-7

java棧、java堆、方法區這3者之間的關係大概就是上面的分析所示。


三、各類異常分析

一、java堆內存溢出錯誤OutOfMemoryError

若是java堆中分配的對象太多,且GC後內存空間仍是不夠用。下面經過循環生成對象來消耗內存空間進行測試。

相關指令:VM Args: -Xms20m -Xmx40m,表示JVM分配的堆內存最小爲20MB,最大爲40MB。

public static void main(String[] args) {
   while (true) {
     List<Object> list = new ArrayList<>(10);
     list.add(new Object());
   }
 }

二、java棧堆棧溢出錯誤StackoverflowError

若是java棧的棧深度大於JVM容許的深度,就會拋出該錯誤。下面經過無限遞歸調用來進行堆棧進行測試。

相關指令:VM Args: -Xss128k,表示JVM分配的棧容量爲128KB。

public class StackOOM {
    
    private int length = 1;
    
    public void stackLeak() {
        length++;
        stackLeak();
    }
    
    public static void main(String[] args) {
        StackOOM stackOOM = new StackOOM();
        stackOOM.stackLeak();
    }
}

jvm-8

相關文章
相關標籤/搜索