日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java8內存模型PermGen Metaspace實例解析

瀏覽:30日期:2022-09-04 08:54:36

一、JVM 內存模型

根據 JVM 規范,JVM 內存共分為虛擬機棧、堆、方法區、程序計數器、本地方法棧五個部分。

Java8內存模型PermGen Metaspace實例解析

1、虛擬機棧:每個線程有一個私有的棧,隨著線程的創建而創建。棧里面存著的是一種叫“棧幀”的東西,每個方法會創建一個棧幀,棧幀中存放了局部變量表(基本數據類型和對象引用)、操作數棧、方法出口等信息。棧的大小可以固定也可以動態擴展。當棧調用深度大于JVM所允許的范圍,會拋出StackOverflowError的錯誤,不過這個深度范圍不是一個恒定的值,我們通過下面這段程序可以測試一下這個結果:

棧溢出測試源碼:

package com.paddx.test.memory; public class StackErrorMock { private static int index = 1; public void call(){ index++; call(); } public static void main(String[] args) { StackErrorMock mock = new StackErrorMock(); try { mock.call(); }catch (Throwable e){ System.out.println('Stack deep : '+index); e.printStackTrace(); } }}

代碼段 1

運行三次,可以看出每次棧的深度都是不一樣的,輸出結果如下。

Java8內存模型PermGen Metaspace實例解析

至于紅色框里的值是怎么出來的,就需要深入到 JVM 的源碼中才能探討,這里不作詳細闡述。

虛擬機棧除了上述錯誤外,還有另一種錯誤,那就是當申請不到空間時,會拋出 OutOfMemoryError。這里有一個小細節需要注意,catch 捕獲的是Throwable,而不是 Exception。因為StackOverflowError 和 OutOfMemoryError 都不屬于 Exception 的子類。

2、本地方法棧:

這部分主要與虛擬機用到的 Native 方法相關,一般情況下, Java 應用程序員并不需要關心這部分的內容。

3、PC 寄存器:

PC 寄存器,也叫程序計數器。JVM支持多個線程同時運行,每個線程都有自己的程序計數器。倘若當前執行的是 JVM 的方法,則該寄存器中保存當前執行指令的地址;倘若執行的是native 方法,則PC寄存器中為空。

4、堆

堆內存是 JVM 所有線程共享的部分,在虛擬機啟動的時候就已經創建。所有的對象和數組都在堆上進行分配。這部分空間可通過 GC 進行回收。當申請不到空間時會拋出 OutOfMemoryError。下面我們簡單的模擬一個堆內存溢出的情況:

package com.paddx.test.memory; import java.util.ArrayList;import java.util.List; public class HeapOomMock { public static void main(String[] args) { List<byte[]> list = new ArrayList<byte[]>(); int i = 0; boolean flag = true; while (flag){ try {i++;list.add(new byte[1024 * 1024]);//每次增加一個1M大小的數組對象 }catch (Throwable e){e.printStackTrace();flag = false;System.out.println('count='+i);//記錄運行的次數 } } }}

代碼段 2

運行上述代碼,輸出結果如下:

Java8內存模型PermGen Metaspace實例解析 

注意,這里我指定了堆內存的大小為16M,所以這個地方顯示的count=14(這個數字不是固定的),至于為什么會是14或其他數字,需要根據 GC 日志來判斷,具體原因會在下篇文章中給大家解釋。

5、方法區:

方法區也是所有線程共享。主要用于存儲類的信息、常量池、方法數據、方法代碼等。方法區邏輯上屬于堆的一部分,但是為了與堆進行區分,通常又叫“非堆”。 關于方法區內存溢出的問題會在下文中詳細探討。

二、PermGen(永久代)

絕大部分 Java 程序員應該都見過 'java.lang.OutOfMemoryError:PermGenspace'這個異常。這里的 “PermGen space”其實指的就是方法區。不過方法區和“PermGen space”又有著本質的區別。前者是 JVM 的規范,而后者則是 JVM 規范的一種實現,并且只有 HotSpot 才有“PermGen space”,而對于其他類型的虛擬機,如 JRockit(Oracle)、J9(IBM) 并沒有“PermGen space”。由于方法區主要存儲類的相關信息,所以對于動態生成類的情況比較容易出現永久代的內存溢出。最典型的場景就是,在 jsp 頁面比較多的情況,容易出現永久代內存溢出。我們現在通過動態生成類來模擬 “PermGenspace”的內存溢出:

package com.paddx.test.memory;public class Test {}

代碼段 3

package com.paddx.test.memory; import java.io.File;import java.net.URL;import java.net.URLClassLoader;import java.util.ArrayList;import java.util.List; public class PermGenOomMock{ public static void main(String[] args) { URL url = null; List<ClassLoader> classLoaderList = new ArrayList<ClassLoader>(); try { url = new File('/tmp').toURI().toURL(); URL[] urls = {url}; while (true){ClassLoader loader = new URLClassLoader(urls);classLoaderList.add(loader);loader.loadClass('com.paddx.test.memory.Test'); } } catch (Exception e) { e.printStackTrace(); } }}

代碼段 4

運行結果如下:

Java8內存模型PermGen Metaspace實例解析

本例中使用的 JDK 版本是 1.7,指定的 PermGen 區的大小為 8M。通過每次生成不同URLClassLoader對象來加載Test類,從而生成不同的類對象,這樣就能看到我們熟悉的'java.lang.OutOfMemoryError:PermGenspace' 異常了。這里之所以采用 JDK 1.7,是因為在 JDK 1.8 中, HotSpot 已經沒有 “PermGen space”這個區間了,取而代之是一個叫做 Metaspace(元空間) 的東西。下面我們就來看看 Metaspace 與 PermGen space 的區別。

三、Metaspace(元空間)

其實,移除永久代的工作從JDK1.7就開始了。JDK1.7中,存儲在永久代的部分數據就已經轉移到了Java Heap或者是 Native Heap。但永久代仍存在于JDK1.7中,并沒完全移除,譬如符號引用(Symbols)轉移到了native heap;字面量(interned strings)轉移到了java heap;類的靜態變量(class statics)轉移到了java heap。我們可以通過一段程序來比較 JDK 1.6 與 JDK 1.7及 JDK 1.8 的區別,以字符串常量為例:

package com.paddx.test.memory; import java.util.ArrayList;import java.util.List; public class StringOomMock { static String base = 'string'; public static void main(String[] args) { List<String> list = new ArrayList<String>(); for (int i=0;i< Integer.MAX_VALUE;i++){ String str = base + base; base = str; list.add(str.intern()); } }}

這段程序以2的指數級不斷的生成新的字符串,這樣可以比較快速的消耗內存。我們通過 JDK 1.6、JDK 1.7 和 JDK 1.8 分別運行:

JDK 1.6 的運行結果:

Java8內存模型PermGen Metaspace實例解析

JDK 1.7的運行結果:

Java8內存模型PermGen Metaspace實例解析

JDK 1.8的運行結果:

Java8內存模型PermGen Metaspace實例解析

從上述結果可以看出,JDK 1.6下,會出現“PermGen Space”的內存溢出,而在 JDK 1.7和 JDK 1.8 中,會出現堆內存溢出,并且 JDK 1.8中 PermSize 和 MaxPermGen 已經無效。因此,可以大致驗證 JDK 1.7 和 1.8 將字符串常量由永久代轉移到堆中,并且 JDK 1.8 中已經不存在永久代的結論。現在我們看看元空間到底是一個什么東西?

元空間的本質和永久代類似,都是對JVM規范中方法區的實現。不過元空間與永久代之間最大的區別在于:元空間并不在虛擬機中,而是使用本地內存。因此,默認情況下,元空間的大小僅受本地內存限制,但可以通過以下參數來指定元空間的大?。?/p>

-XX:MetaspaceSize,初始空間大小,達到該值就會觸發垃圾收集進行類型卸載,同時GC會對該值進行調整:如果釋放了大量的空間,就適當降低該值;如果釋放了很少的空間,那么在不超過MaxMetaspaceSize時,適當提高該值。-XX:MaxMetaspaceSize,最大空間,默認是沒有限制的。

除了上面兩個指定大小的選項以外,還有兩個與 GC 相關的屬性:-XX:MinMetaspaceFreeRatio,在GC之后,最小的Metaspace剩余空間容量的百分比,減少為分配空間所導致的垃圾收集-XX:MaxMetaspaceFreeRatio,在GC之后,最大的Metaspace剩余空間容量的百分比,減少為釋放空間所導致的垃圾收集

現在我們在 JDK 8下重新運行一下代碼段 4,不過這次不再指定 PermSize 和 MaxPermSize。而是指定 MetaSpaceSize 和 MaxMetaSpaceSize的大小。輸出結果如下:

Java8內存模型PermGen Metaspace實例解析

從輸出結果,我們可以看出,這次不再出現永久代溢出,而是出現了元空間的溢出。

四、總結

通過上面分析,大家應該大致了解了 JVM 的內存劃分,也清楚了 JDK 8 中永久代向元空間的轉換。不過大家應該都有一個疑問,就是為什么要做這個轉換?所以,最后給大家總結以下幾點原因:

1、字符串存在永久代中,容易出現性能問題和內存溢出。

2、類及方法的信息等比較難確定其大小,因此對于永久代的大小指定比較困難,太小容易出現永久代溢出,太大則容易導致老年代溢出。

3、永久代會為 GC 帶來不必要的復雜度,并且回收效率偏低。

4、Oracle 可能會將HotSpot 與 JRockit 合二為一。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日韩免费久久| 日韩欧美激情| 欧美1区二区| 国产精品久久久久久久久免费高清| 亚洲精品一级| 日韩精品视频网| 欧美亚洲一级| 精品一区二区三区的国产在线观看 | 国产精品一区二区三区美女 | 国产福利一区二区三区在线播放| 国产精品15p| 精品国产99| 久久精品亚洲人成影院| 九九综合九九| 伊人国产精品| 国产精品一区二区99| 风间由美中文字幕在线看视频国产欧美| 精品一区视频| 欧美1区2区3区| 亚洲综合婷婷| 久久精品国产亚洲aⅴ| 视频小说一区二区| 伊人精品久久| 欧美三区不卡| 国产精品久久久久久久久妇女| 久久精品亚洲欧美日韩精品中文字幕| 欧美精品一线| 日韩高清在线观看一区二区| 麻豆中文一区二区| 久久精品欧美一区| 日韩美女精品| 国产91在线精品| 亚洲一区欧美激情| 国产日产精品一区二区三区四区的观看方式 | 高清一区二区三区av| 欧美日韩在线二区| 香蕉久久一区| 国产成人精品一区二区三区视频 | 蜜臀av一区二区在线免费观看 | 亚洲激情欧美| 国产日产精品_国产精品毛片 | 国产欧美另类| 日韩一区二区三区在线免费观看| 在线视频精品| 久久一区欧美| 午夜在线精品偷拍| 国产乱码精品| 国产精品99免费看| 日本91福利区| 少妇久久久久| 日本欧美在线| 宅男在线一区| 国产精品多人| 久久国产成人| 四季av一区二区凹凸精品| 午夜一级在线看亚洲| 福利一区和二区| 在线精品观看| 色婷婷狠狠五月综合天色拍| 亚洲a级精品| 成人久久一区| 欧美日韩亚洲一区在线观看| 欧洲精品一区二区三区| 日本成人手机在线| 精品一区在线| 精品国产aⅴ| 日韩精品一二区| 国产一区2区在线观看| 亚洲色图网站| 欧美99久久| 中文字幕在线免费观看视频| 青青草91久久久久久久久| 亚洲二区在线| 国内自拍视频一区二区三区| 中文字幕成人| 国产在线成人| 欧美91在线| 亚洲精品四区| 一级欧洲+日本+国产| 国产精品黑丝在线播放| 国产欧美日韩在线观看视频 | 久久国产精品色av免费看| 欧美日韩国产高清| 久久电影tv| 美女视频黄免费的久久| 视频一区日韩| 99综合视频| 午夜av成人| 福利精品在线| 国产精品天堂蜜av在线播放| 中文字幕一区二区av| 亚洲欧美一区在线| 久久久9色精品国产一区二区三区| 精品视频在线观看网站| 青青草伊人久久| 中文字幕视频精品一区二区三区| 婷婷久久一区| 国产99久久| 桃色av一区二区| 国产一区二区三区不卡视频网站 | 国产图片一区| 欧美专区一区| 久久激情av| 日韩一区二区三区免费视频| 国产亚洲亚洲| 国产亚洲亚洲| 亚洲免费精品| 中文一区二区| 中文字幕一区二区三区四区久久| 亚洲激情欧美| 久久香蕉精品| 日本 国产 欧美色综合| 亚洲综合三区| 日韩中文字幕区一区有砖一区| 尹人成人综合网| 亚洲一级在线| 午夜在线视频一区二区区别| 欧美日韩国产综合网| 中文另类视频| 久久在线免费| 在线综合视频| 亚洲尤物av| 日本视频中文字幕一区二区三区| 日韩高清不卡一区二区| 国产区精品区| 荡女精品导航| 99久久精品费精品国产| 激情欧美日韩一区| 91久久在线| 中文字幕成人| 国产精品香蕉| 久久电影tv| 一区在线免费| 在线观看一区| 国产精品二区影院| 欧美一级鲁丝片| 蜜桃视频在线观看一区二区| 日韩av网站在线免费观看| 国产精品第一国产精品| 国产色播av在线| 国产二区精品| 亚洲日韩中文字幕一区| 国产精品红桃| 亚洲不卡系列| 视频一区中文字幕| 国产日韩精品视频一区二区三区| 久久中文在线| 亚洲成人va| 91久久中文| 日韩免费精品| 高清一区二区| 亚洲精品网址| 久久精品超碰| 亚洲播播91| 蜜桃视频在线观看一区| 欧美国产另类| 午夜精品婷婷| 欧美日韩一区二区三区在线电影| а√天堂8资源在线| 日韩视频一区| 欧美精品中文字幕亚洲专区| 国产精选在线| 亚洲一区网站| 欧美一区成人| 日韩欧美一区二区三区在线观看 | 久久国产人妖系列| 婷婷激情一区| 免费在线观看一区二区三区| 日韩一区二区三区在线看| 精品免费av| 伊人精品在线| 欧美激情精品| 欧美.日韩.国产.一区.二区| 91精品日本| 欧美日韩中文一区二区| 日韩高清一级| 久久伦理在线| 国产日韩免费| 国精品一区二区| 国产午夜一区| 欧美国产91| 欧美国产精品| 午夜在线精品偷拍| 精品理论电影在线| 亚洲视频国产| 私拍精品福利视频在线一区| 日韩极品在线观看| 欧美日韩尤物久久| 国产一级成人av| 在线视频精品| 97精品国产福利一区二区三区| 亚洲精品一二三**| 99视频精品全国免费| 国产精品一区二区av交换| 日韩午夜av| 欧美成人a交片免费看| 国产伦精品一区二区三区千人斩| 免费精品国产的网站免费观看| 欧美日韩在线精品一区二区三区激情综合| 欧洲亚洲一区二区三区|