电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術(shù)文章
文章詳情頁

Java8內(nèi)存模型PermGen Metaspace實(shí)例解析

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

一、JVM 內(nèi)存模型

根據(jù) JVM 規(guī)范,JVM 內(nèi)存共分為虛擬機(jī)棧、堆、方法區(qū)、程序計(jì)數(shù)器、本地方法棧五個(gè)部分。

Java8內(nèi)存模型PermGen Metaspace實(shí)例解析

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

棧溢出測試源碼:

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

運(yùn)行三次,可以看出每次棧的深度都是不一樣的,輸出結(jié)果如下。

Java8內(nèi)存模型PermGen Metaspace實(shí)例解析

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

虛擬機(jī)棧除了上述錯(cuò)誤外,還有另一種錯(cuò)誤,那就是當(dāng)申請不到空間時(shí),會(huì)拋出 OutOfMemoryError。這里有一個(gè)小細(xì)節(jié)需要注意,catch 捕獲的是Throwable,而不是 Exception。因?yàn)镾tackOverflowError 和 OutOfMemoryError 都不屬于 Exception 的子類。

2、本地方法棧:

這部分主要與虛擬機(jī)用到的 Native 方法相關(guān),一般情況下, Java 應(yīng)用程序員并不需要關(guān)心這部分的內(nèi)容。

3、PC 寄存器:

PC 寄存器,也叫程序計(jì)數(shù)器。JVM支持多個(gè)線程同時(shí)運(yùn)行,每個(gè)線程都有自己的程序計(jì)數(shù)器。倘若當(dāng)前執(zhí)行的是 JVM 的方法,則該寄存器中保存當(dāng)前執(zhí)行指令的地址;倘若執(zhí)行的是native 方法,則PC寄存器中為空。

4、堆

堆內(nèi)存是 JVM 所有線程共享的部分,在虛擬機(jī)啟動(dòng)的時(shí)候就已經(jīng)創(chuàng)建。所有的對象和數(shù)組都在堆上進(jìn)行分配。這部分空間可通過 GC 進(jìn)行回收。當(dāng)申請不到空間時(shí)會(huì)拋出 OutOfMemoryError。下面我們簡單的模擬一個(gè)堆內(nèi)存溢出的情況:

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]);//每次增加一個(gè)1M大小的數(shù)組對象 }catch (Throwable e){e.printStackTrace();flag = false;System.out.println('count='+i);//記錄運(yùn)行的次數(shù) } } }}

代碼段 2

運(yùn)行上述代碼,輸出結(jié)果如下:

Java8內(nèi)存模型PermGen Metaspace實(shí)例解析 

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

5、方法區(qū):

方法區(qū)也是所有線程共享。主要用于存儲(chǔ)類的信息、常量池、方法數(shù)據(jù)、方法代碼等。方法區(qū)邏輯上屬于堆的一部分,但是為了與堆進(jìn)行區(qū)分,通常又叫“非堆”。 關(guān)于方法區(qū)內(nèi)存溢出的問題會(huì)在下文中詳細(xì)探討。

二、PermGen(永久代)

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

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

運(yùn)行結(jié)果如下:

Java8內(nèi)存模型PermGen Metaspace實(shí)例解析

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

三、Metaspace(元空間)

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

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的指數(shù)級不斷的生成新的字符串,這樣可以比較快速的消耗內(nèi)存。我們通過 JDK 1.6、JDK 1.7 和 JDK 1.8 分別運(yùn)行:

JDK 1.6 的運(yùn)行結(jié)果:

Java8內(nèi)存模型PermGen Metaspace實(shí)例解析

JDK 1.7的運(yùn)行結(jié)果:

Java8內(nèi)存模型PermGen Metaspace實(shí)例解析

JDK 1.8的運(yùn)行結(jié)果:

Java8內(nèi)存模型PermGen Metaspace實(shí)例解析

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

元空間的本質(zhì)和永久代類似,都是對JVM規(guī)范中方法區(qū)的實(shí)現(xiàn)。不過元空間與永久代之間最大的區(qū)別在于:元空間并不在虛擬機(jī)中,而是使用本地內(nèi)存。因此,默認(rèn)情況下,元空間的大小僅受本地內(nèi)存限制,但可以通過以下參數(shù)來指定元空間的大小:

-XX:MetaspaceSize,初始空間大小,達(dá)到該值就會(huì)觸發(fā)垃圾收集進(jìn)行類型卸載,同時(shí)GC會(huì)對該值進(jìn)行調(diào)整:如果釋放了大量的空間,就適當(dāng)降低該值;如果釋放了很少的空間,那么在不超過MaxMetaspaceSize時(shí),適當(dāng)提高該值。-XX:MaxMetaspaceSize,最大空間,默認(rèn)是沒有限制的。

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

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

Java8內(nèi)存模型PermGen Metaspace實(shí)例解析

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

四、總結(jié)

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

1、字符串存在永久代中,容易出現(xiàn)性能問題和內(nèi)存溢出。

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

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

4、Oracle 可能會(huì)將HotSpot 與 JRockit 合二為一。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 碳纤维复合材料制品生产定制工厂订制厂家-凯夫拉凯芙拉碳纤维手机壳套-碳纤维雪茄盒外壳套-深圳市润大世纪新材料科技有限公司 | 模具ERP_模具管理系统_模具mes_模具进度管理_东莞市精纬软件有限公司 | 必胜高考网_全国高考备考和志愿填报信息平台 | 宁夏档案密集柜,智能密集柜,电动手摇密集柜-盛隆柜业宁夏档案密集柜厂家 | 南京兰江泵业有限公司-水解酸化池潜水搅拌机-絮凝反应池搅拌机-好氧区潜水推进器 | 成都办公室装修-办公室设计-写字楼装修设计-厂房装修-四川和信建筑装饰工程有限公司 | 胜为光纤光缆_光纤跳线_单模尾纤_光纤收发器_ODF光纤配线架厂家直销_北京睿创胜为科技有限公司 - 北京睿创胜为科技有限公司 | 贵州科比特-防雷公司厂家提供贵州防雷工程,防雷检测,防雷接地,防雷设备价格,防雷产品报价服务-贵州防雷检测公司 | 鲸鱼视觉 -数字展厅多媒体互动展示制作公司 | 螺旋压榨机-刮泥机-潜水搅拌机-电动泥斗-潜水推流器-南京格林兰环保设备有限公司 | 石家庄小程序开发_小程序开发公司_APP开发_网站制作-石家庄乘航网络科技有限公司 | 塑胶地板-商用PVC地板-pvc地板革-安耐宝pvc塑胶地板厂家 | 涡街流量计_LUGB智能管道式高温防爆蒸汽温压补偿计量表-江苏凯铭仪表有限公司 | 粉末冶金-粉末冶金齿轮-粉末冶金零件厂家-东莞市正朗精密金属零件有限公司 | 合肥钣金加工-安徽激光切割加工-机箱机柜加工厂家-合肥通快 | 锌合金压铸-铝合金压铸厂-压铸模具-冷挤压-誉格精密压铸 | 软文发布-新闻发布推广平台-代写文章-网络广告营销-自助发稿公司媒介星 | 骁龙云呼电销防封号系统-axb电销平台-外呼稳定『免费试用』 | 杭州标识标牌|文化墙|展厅|导视|户内外广告|发光字|灯箱|铭阳制作公司 - 杭州标识标牌|文化墙|展厅|导视|户内外广告|发光字|灯箱|铭阳制作公司 | 成都竞价托管_抖音代运营_网站建设_成都SEM外包-成都智网创联网络科技有限公司 | 陕西安玻璃自动感应门-自动重叠门-磁悬浮平开门厂家【捷申达门业】 | UV固化机_UVLED光固化机_UV干燥机生产厂家-上海冠顶公司专业生产UV固化机设备 | 蓄电池在线监测系统|SF6在线监控泄露报警系统-武汉中电通电力设备有限公司 | 注塑模具_塑料模具_塑胶模具_范仕达【官网】_东莞模具设计与制造加工厂家 | 匀胶机旋涂仪-声扫显微镜-工业水浸超声-安赛斯(北京)科技有限公司 | 上海橡胶接头_弹簧减震器_金属软接头厂家-上海淞江集团 | 振动时效_振动时效仪_超声波冲击设备-济南驰奥机电设备有限公司 北京宣传片拍摄_产品宣传片拍摄_宣传片制作公司-现像传媒 | 碳纤维布-植筋胶-灌缝胶-固特嘉加固材料公司 | 智能交通网_智能交通系统_ITS_交通监控_卫星导航_智能交通行业 | 免费B2B信息推广发布平台 - 推发网| 油液红外光谱仪-油液监测系统-燃油嗅探仪-上海冉超光电科技有限公司 | 螺旋丝杆升降机-SWL蜗轮-滚珠丝杆升降机厂家-山东明泰传动机械有限公司 | 数控专用机床,专用机床,自动线,组合机床,动力头,自动化加工生产线,江苏海鑫机床有限公司 | 恒湿机_除湿加湿一体机_恒湿净化消毒一体机厂家-杭州英腾电器有限公司 | 神马影院-实时更新秒播 | 手机游戏_热门软件app下载_好玩的安卓游戏下载基地-吾爱下载站 | 低气压试验箱_高低温低气压试验箱_低气压实验箱 |林频试验设备品牌 | 菲希尔FISCHER测厚仪-铁素体检测仪-上海吉馨实业发展有限公司 | 顺景erp系统_erp软件_erp软件系统_企业erp管理系统-广东顺景软件科技有限公司 | 耐驰泵阀管件制造-耐驰泵阀科技(天津)有限公司| SMC-ASCO-CKD气缸-FESTO-MAC电磁阀-上海天筹自动化设备官网 |