麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 學院 > 開發設計 > 正文

JVM 各區域的用途以及潛在出現異常的示例

2019-11-11 07:14:28
字體:
來源:轉載
供稿:網友

學習java的同學注意了!!! 學習過程中遇到什么問題或者想獲取學習資源的話,歡迎加入Java學習交流群,群號碼:183993990  我們一起學Java!

程序計數器

用于給字節碼解釋器來選取嚇一跳需要執行的字節碼指令。每個線程有一個獨立的程序計數器去,且各個線程之間互不影響。如果線程正在執行一個Java方法,這個計數器記錄的是正在執行的虛擬機字節碼指令的內存地址;如果執行的是Native方法。在計數器為Undefined。此區域是JVM規范中唯一一個不存在OOM的區域

虛擬機棧(局部變量空間)

存放編譯器可知的各種基本數據類型(boolean、byte、char、short、int、float、long、double)、對象應用(reference)。64位的double、long占用2個槽。內存空間在編譯期間就能確定,當進入一個方式時,這個方法需要分配的局部變量空間是完全確定的,通過-Xss設置內存容量

異常狀況:

StackOverflowError 棧深度大于虛擬機所允許的深度

OOM 如果虛擬機棧可以動態擴展(當前大部分Java虛擬機都可以動態擴展,只不過Java虛擬機規范中的也允許固定長度的虛擬機棧),如果擴展是無法申請到足夠的內存

在單個線程下,無論是猶豫棧幀太大還是虛擬機棧容量太小,當內存無法分配的時候,虛擬機拋出的都是StackOverflowError

/*** VM Args:-Xss128k* * stack length:2402 Exception in thread "main" java.lang.StackOverflowError*/public class JavaVMStackSOF {PRivate int stackLength = 1;public void stackLeak() {    stackLength++;    stackLeak();}public static void main(String[] args) throws Throwable {    JavaVMStackSOF oom = new JavaVMStackSOF();    try {        oom.stackLeak();    } catch (Throwable e) {        System.out.println("stack length:" + oom.stackLength);        throw e;    }}}

如果測試時不限于單線程,通過不斷的建立線程的方式到是可以產生內存溢出的異常。但是這樣產生的內存溢出異常與占空間是否足夠大并不存在任何聯系,或者準確的說,在這種情況下,為每個線程的棧分配的內存越大,反而越容易產生內存溢出的異常。

因為操作系統分配給每個進程的內存是有限制的,譬如32的window限制為2GB。此測試是通過創建大量的線程。每個線程占用棧內存分配大量的內存,導致系統沒有足夠的內存。使其不能自動擴展

/** * VM Args:-Xss2M (這時候不妨設大些) * * java.lang.OutOfMemoryError:unable to create new native thread */public class JavaVMStackOOM {       private void dontStop() {              while (true) {              }       }       public void stackLeakByThread() {              while (true) {                     Thread thread = new Thread(new Runnable() {                            @Override                            public void run() {                                   dontStop();                            }                     });                     thread.start();              }       }       public static void main(String[] args) throws Throwable {              JavaVMStackOOM oom = new JavaVMStackOOM();              oom.stackLeakByThread();       }}

本地方法棧

跟虛擬機棧類似,只是一個是虛擬機執行Java方法,一個是執行Native方法

異常狀況:

StackOverflowError 棧深度大于虛擬機所允許的深度

OOM

Java堆

線程共享的一塊內存區域,從內存回收角度來看,基本都采用分代收集算法,所以分為新生代、老年代。再細致一點可以分為Eden空間、From Survivor空間、To Survivor空間等。-Xmx -Xms控制堆空間大小

異常狀況:

1.OOM 堆無法擴展時

/** * VM Args:-Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError *  * java.lang.OutOfMemoryError: Java heap space */public class HeapOOM {    static class OOMObject {    }    public static void main(String[] args) {        List<OOMObject> list = new ArrayList<OOMObject>();        while (true) {            list.add(new OOMObject());        }    }}

方法區

線程間共享。存儲已經被虛擬機加載的類信息、常量、靜態變量、即時編輯器編譯后的代碼等數據,在HotSpot虛擬機中可以稱為永生代。

運行時常量在1.6及之前是方法區的一部分(String.intern()動態加入常量池) -XX:MaxPermSize控制大小。在JDK1.7及之后的版本是在Java堆中開辟的一塊內存

異常狀況:

OOM

/*** 需要在JDK1.6上才能復現,JDK1.7及之后版本的JVM已經將運行時常量池從方法區中移了出來,在Java 堆(Heap)中開辟了一塊區域存放運行時常量池。* 在JDK1.7上運行的效果則會一直執行,直到堆內存使用完畢* VM Args:-XX:PermSize=10M -XX:MaxPermSize=10M ** java.lang.OutOfMemoryError:PermGen space*/public class RuntimeConstantPoolOOM {public static void main(String[] args) {    // 使用List保持著常量池引用,避免Full GC回收常量池行為    List<String> list = new ArrayList<String>();    // 10MB的PermSize在integer范圍內足夠產生OOM了    int i = 0;    while (true) {        list.add(String.valueOf(i++).intern());    }}}
/*** VM Args: -XX:PermSize=10M -XX:MaxPermSize=10M* java.lang.OutOfMemoryError:PermGen space* 一直創建動態類*/public class JavaMethodAreaOOM {public static void main(String[] args) {    while (true) {        Enhancer enhancer = new Enhancer();        enhancer.setSuperclass(OOMObject.class);        enhancer.setUseCache(false);        enhancer.setCallback(new MethodInterceptor() {            public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {                return proxy.invokeSuper(obj, args);            }        });        enhancer.create();    }}static class OOMObject {}}

直接內存(不屬于虛擬機運行時的數據區的一部分)

NIO可以使用Native函數庫直接分配對外的內存,然后通過存儲在Java對中的DirectByteBuffer對象作為這塊內存的引用進行操作。受限于機器物理內存,可以通過-XX:MaxDirectMemorySize制定,如果不制定,默認與Java堆最大值(-Xmx)一樣

異常狀況:

1.OOM

/** * VM Args:-Xmx20M -XX:MaxDirectMemorySize=10M *  * java.lang.OutOfMemoryError */public class DirectMemoryOOM {    private static final int _1MB = 1024 * 1024;    public static void main(String[] args) throws Exception {        Field unsafeField = Unsafe.class.getDeclaredFields()[0];        unsafeField.setaccessible(true);        Unsafe unsafe = (Unsafe) unsafeField.get(null);        while (true) {            unsafe.allocateMemory(_1MB);        }    }}

學習Java的同學注意了!!! 學習過程中遇到什么問題或者想獲取學習資源的話,歡迎加入Java學習交流群,群號碼:183993990  我們一起學Java!


上一篇:對json 的總結和理解

下一篇:抽象類

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 羞羞视频免费观看网站 | 国产精品1区2区 | 九九热精品视频在线 | 杏美月av| 久草在线视频新 | 美女黄污视频 | lutube成人福利在线观看 | 亚洲精品永久视频 | 久草手机在线 | 成人做爰高潮片免费视频美国 | 欧美黑人xx | 久久久免费观看完整版 | 97超级碰碰人国产在线观看 | 成人黄视频在线观看 | 一级黄色免费大片 | 午夜视频在线观看免费视频 | 久久一区国产 | 成年人黄色免费网站 | 欧美a在线观看 | 草莓福利社区在线 | 欧美精品成人一区二区在线观看 | 99精品视频一区二区三区 | 黄色毛片免费视频 | 九九热在线免费观看视频 | 精品久久久久久久久久久久包黑料 | 最近国产中文字幕 | 亚洲午夜精品视频 | 免费的毛片 | 1级黄色毛片 | 亚洲精品自在在线观看 | 久久久中文 | 国产免费观看a大片的网站 欧美成人一级 | 性少妇videosexfreexxx片 | 色妇视频 | jizzjizz中国少妇中文 | 国产精品九九久久一区hh | 国产电影精品久久 | 日本免费aaa观看 | 国产成人精品午夜视频' | 男女污污视频网站 | 精品国产三级a |