本篇內(nèi)容介紹了“Java內(nèi)存區(qū)域與內(nèi)存溢出異常知識講解”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!
創(chuàng)新互聯(lián)專業(yè)為企業(yè)提供龍里網(wǎng)站建設(shè)、龍里做網(wǎng)站、龍里網(wǎng)站設(shè)計、龍里網(wǎng)站制作等企業(yè)網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計與制作、龍里企業(yè)網(wǎng)站模板建站服務(wù),10多年龍里做網(wǎng)站經(jīng)驗,不只是建網(wǎng)站,更提供有價值的思路和整體網(wǎng)絡(luò)服務(wù)。
在開始講解之前, 需要先明確關(guān)于 JVM
的一些基本概念
我們都知道, Java
是一個跨平臺的語言, Java
跨平臺的基本支撐其實就是 JVM
對操作系統(tǒng)底層細(xì)節(jié)的屏蔽, 相當(dāng)于加了一個中間層(計算機中的任何問題都可以加一個中間層解決~), Java
不再像 C/C++
等語言一樣直接翻譯為針對特殊平臺的機器碼, 而是翻譯為字節(jié)碼, 也即是我們的 class
文件, 下圖大概可以比較簡明的概括了~; 字節(jié)碼就相當(dāng)于 Java
世界中的匯編, 而 JVM
則不是跨平臺的, 只是不同平臺的 JVM
都能識別和運行標(biāo)準(zhǔn)格式的字節(jié)碼文件而已
關(guān)于 JVM
運行 class
文件, 我覺得下圖已經(jīng)可以比較準(zhǔn)確的表達(dá)了
我們下面要講的就是 Runtime Data Area
部分
JVM
會在執(zhí)行 Java
程序的時候把它所管理的內(nèi)存劃分為若干個不同的數(shù)據(jù)區(qū), 如下:
線程私有
指向下一條需要執(zhí)行的字節(jié)碼指令; 如果線程正在執(zhí)行一個 Java
方法, 該計數(shù)器記錄的是正在執(zhí)行的虛擬機字節(jié)碼指令的地址; 如果正在執(zhí)行 Native
方法, 該計數(shù)器值則為空( Undefined
)
該區(qū)域是是唯一一個在 Java
虛擬機中沒有規(guī)定任何 OutOfMemoryError
情況的區(qū)域
線程私有
描述 Java
方法執(zhí)行的內(nèi)存模型, 每個方法調(diào)用就對應(yīng)著一個棧幀的入棧和出棧; 一個棧幀里面存儲了局部變量表, 操作數(shù)棧, 動態(tài)鏈接, 方法出口等信息
局部變量表存儲了編譯器可知的各種基本數(shù)據(jù)類型, 對象引用, returnAddress
; 局部變量表的大小在編譯期間即可確定, 運行期間大小不變
StackOverflowError : 線程請求棧深度大于虛擬機允許深度
異常示例代碼:
public class JavaVMStackSOF { private int stackLength = 1; public void stackLeak() { stackLength++; stackLeak(); } public static void main(String[] args) { JavaVMStackSOF sof = new JavaVMStackSOF(); try { sof.stackLeak(); } catch (Throwable e) { System.out.println("Stack Length: " + sof.stackLength); throw e; } } }
OutOfMemoryError : 虛擬機棧動態(tài)擴展時無法申請到足夠內(nèi)存
異常示例代碼:
public class JavaVMStackOOM { private void dontStop() { while (true) { } } public void stackLeakByThread() { while (true) { new Thread(new Runnable() { @Override public void run() { dontStop(); } }).start(); } } public static void main(String[] args) { JavaVMStackOOM oom = new JavaVMStackOOM(); oom.stackLeakByThread(); } }
注: 由于操作系統(tǒng)分配給每個進程的內(nèi)存空間是有限制的, 所以如果是由于建立過多的線程導(dǎo)致內(nèi)存溢出, 在不能減少線程數(shù)或者更換 64
位虛擬機的情況下, 可以選擇通過減少最大堆和減少棧容量來換取更多的線程
線程私有
和虛擬機棧類似, 只是本地方法棧提供的是 Native
方法服務(wù)
StackOverflowError
和 OutOfMemoryError
線程共享
垃圾收集管理的主要區(qū)域
幾乎所有的對象實例都在這里分配
OutOfMemoryError
異常示例:
public class JavaVMHeapOOM { static class HeapOOM { } public static void main(String[] args) { List<HeapOOM> list = new ArrayList(); while (true) { list.add(new HeapOOM()); } } }
線程共享
該區(qū)域的垃圾回收目標(biāo)主要是針對常量池的回收和對類型的卸載
存儲已被虛擬機加載的類信息, 常量, 靜態(tài)變量, 即使編譯器編譯后的代碼等數(shù)據(jù)
運行時常量池是方法區(qū)的一部分, 但是 JDK6
之后, 常量池被放入了堆中;
Class
文件中也有常量池部分, 即編譯期生成的各種字面量和符號引用, 這部分將在類加載后進入方法區(qū)的運行時常量池中, 此外還會把翻譯出來的直接引用也存儲在運行時常量池中
運行時常量池相對于 Class
文件常量池的另外一個最重要的特征是具備動態(tài)性, 即運行期間也可以將新的常量放入池中, 比如 String
的 intern()
方法
String.intern()
作用是: 如果字符串常量池中已經(jīng)包含一個等于此 String
對象的字符串, 則返回代表池中這個字符串的 String
對象; 否則, 將此 String
對象包含的字符串添加到常量池中, 并且返回此 String
對象的引用
同樣, 收方法區(qū)的限制, 當(dāng)常量池?zé)o法再申請到內(nèi)存時會拋出 OutOfMemoryError
OutOfMemoryError
: 方法區(qū)無法滿足內(nèi)存分配需求
異常示例:
public class RuntimeConstantPoolOOM { public static void main(String[] args) { List<String> list = new ArrayList<>(); int i = 0; while (true) { list.add(String.valueOf(i++).intern()); } } }
直接內(nèi)存不是虛擬機運行時數(shù)據(jù)區(qū)的一部分, 但是也被頻繁使用, 如: 在 JDK1.4
中新加入了 NIO
類, 引入了一種基于通道( Chanel
)和緩沖區(qū)( Buffer
)的 I/O
方式, 它可以使用 Native
函數(shù)庫直接分配堆外內(nèi)存, 然后通過一個存儲在 Java
堆中的 DirectByteBuffer
對象作為這塊內(nèi)存的引用進行操作, 避免了在 Java
堆和 Native
堆中來回復(fù)制數(shù)據(jù), 提高性能
同樣會產(chǎn)生 OutOfMemoryError
“Java內(nèi)存區(qū)域與內(nèi)存溢出異常知識講解”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!
文章題目:Java內(nèi)存區(qū)域與內(nèi)存溢出異常知識講解
文章分享:http://chinadenli.net/article14/gjdoge.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供電子商務(wù)、外貿(mào)網(wǎng)站建設(shè)、手機網(wǎng)站建設(shè)、云服務(wù)器、商城網(wǎng)站、服務(wù)器托管
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)