自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

JVM的內(nèi)存溢出異常

開發(fā) 后端
在Java虛擬機(jī)規(guī)范的描述中,除了PC(程序計(jì)數(shù)器)寄存器外,虛擬機(jī)內(nèi)存的其他幾個(gè)運(yùn)行時(shí)區(qū)域都有發(fā)生OutOfMemoryError異常的可能。當(dāng)發(fā)生OutOfMemoryError異常時(shí),無(wú)法用try...catch捕捉。

在Java虛擬機(jī)規(guī)范的描述中,除了PC(程序計(jì)數(shù)器)寄存器外,虛擬機(jī)內(nèi)存的其他幾個(gè)運(yùn)行時(shí)區(qū)域都有發(fā)生OutOfMemoryError異常的可能。當(dāng)發(fā)生OutOfMemoryError異常時(shí),無(wú)法用try...catch捕捉。

在開始講解之前,在這里先簡(jiǎn)單介紹下虛擬機(jī)啟動(dòng)相關(guān)的一些內(nèi)存設(shè)置參數(shù)。因?yàn)镺utOfMemoryError異常發(fā)生,與這些參數(shù)的設(shè)置密切相關(guān)。

舉例說(shuō)明含義:

-Xss128k

每個(gè)線程的java棧大小,一個(gè)線程java棧所有棧幀大小總和***允許的尺寸128k。

-Xms128m

表示JVM Heap(堆內(nèi)存)最小尺寸128MB,初始分配

-Xmx512m

表示JVM Heap(堆內(nèi)存)***允許的尺寸256MB,按需分配。

-XX:PermSize=20M

設(shè)置方法區(qū)的初始大小

-XX:MaxPermSize=30M

設(shè)置方法區(qū)的***值

Java棧溢出

在Java虛擬機(jī)規(guī)范中,對(duì)這個(gè)區(qū)域規(guī)定了兩種異常狀況:StackOverflowError和OutOfMemoryError異常。

1.StackOverflowError異常

每當(dāng)java程序代碼啟動(dòng)一個(gè)新線程時(shí),Java虛擬機(jī)都會(huì)為它分配一個(gè)Java棧。Java棧以幀為單位保存線程的運(yùn)行狀態(tài)。當(dāng)線程調(diào)用java方法時(shí),虛擬機(jī)壓入一個(gè)新的棧幀到該線程的java棧中。只要這個(gè)方法還沒(méi)有返回,它就一直存在。如果線程的方法嵌套調(diào)用層次太多(如遞歸調(diào)用),隨著java棧中幀的逐漸增多,最終會(huì)由于該線程java棧中所有棧幀大小總和大于-Xss設(shè)置的值,而產(chǎn)生StackOverflowError內(nèi)存溢出異常。例子如下:

  1. /**  
  2.   * VM Args: -Xss128k  
  3.   */ 
  4.  public class Test {  
  5.        
  6.      private int count = 0;  
  7.    
  8.      public static void main(String[] args) {  
  9.          new Test().method();  
  10.      }  
  11.        
  12.      public void method() {  
  13.          System.out.println(++count);  
  14.          method();  
  15.      }  
  16.    
  17.  } 

-Xss為128k。其中的一次測(cè)試結(jié)果為,當(dāng)count的值累加到2312時(shí),發(fā)生如下異常:

  1. Exception in thread "main" java.lang.StackOverflowError  
  2.     at sun.nio.cs.UTF_8.updatePositions(UTF_8.java:58)  
  3.     at sun.nio.cs.UTF_8$Encoder.encodeArrayLoop(UTF_8.java:392)  
  4.     at sun.nio.cs.UTF_8$Encoder.encodeLoop(UTF_8.java:447)  
  5.     at java.nio.charset.CharsetEncoder.encode(CharsetEncoder.java:544)  
  6.     at sun.nio.cs.StreamEncoder.implWrite(StreamEncoder.java:252)  
  7.     at sun.nio.cs.StreamEncoder.write(StreamEncoder.java:106)  
  8.     at java.io.OutputStreamWriter.write(OutputStreamWriter.java:190)  
  9.     at java.io.BufferedWriter.flushBuffer(BufferedWriter.java:111)  
  10.     at java.io.PrintStream.write(PrintStream.java:476)  
  11.     at java.io.PrintStream.print(PrintStream.java:547)  
  12.     at java.io.PrintStream.println(PrintStream.java:686)  
  13.     at jvm.Test.method(Test.java:17

修改-Xss為1280k。其中的一次測(cè)試結(jié)果為,當(dāng)count的值累加到26888時(shí),發(fā)生StackOverflowError異常。隨著-Xss參數(shù)值的增大,可以嵌套的方法調(diào)用層次也相應(yīng)增加。

綜上所述,StackOverflowError異常是由于方法調(diào)用的層次太深,最終導(dǎo)致為某個(gè)線程分配的所有棧幀大小總和大于-Xss設(shè)置的值,從而發(fā)生StackOverflowError異常。

2.OutOfMemoryError異常

java程序代碼啟動(dòng)一個(gè)新線程時(shí),沒(méi)有足夠的內(nèi)存空間為該線程分配java棧(一個(gè)線程java棧的大小由-Xss參數(shù)確定),jvm則拋出OutOfMemoryError異常。例子如下:

  1. /**  
  2.   * VM Args: -Xss128k  
  3.   */ 
  4.  public class Test {  
  5.        
  6.      public static void main(String[] args) {  
  7.          int count = 0;  
  8.          while (true) {  
  9.              Thread thread = new Thread(new Runnable() {  
  10.                  public void run() {  
  11.                      while (true) {  
  12.                          try {  
  13.                              Thread.sleep(5000);  
  14.                          } catch (Exception e) {}  
  15.                      }  
  16.                  }  
  17.              });    
  18.              thread.start();  
  19.              System.out.println(++count);  
  20.          }  
  21.      }  
  22.    
  23.  } 

-Xss為128k。其中的一次測(cè)試結(jié)果為,當(dāng)count的值累加到11887時(shí),發(fā)生如下異常:

  1. Exception in thread "main" java.lang.OutOfMemoryError: unable to create new native thread  
  2.     at java.lang.Thread.start0(Native Method)  
  3.     at java.lang.Thread.start(Thread.java:640)  
  4.     at jvm.Test.main(Test.java:20

修改-Xss為1280k。其中的一次測(cè)試結(jié)果為,當(dāng)count的值累加到1270時(shí),發(fā)生OutOfMemoryError異常。隨著-Xss參數(shù)值的增大,java程序可以創(chuàng)建的總線程數(shù)越少。

Java堆溢出

Java堆用于儲(chǔ)存對(duì)象實(shí)例。當(dāng)需要為對(duì)象實(shí)例分配內(nèi)存,而堆的內(nèi)存占用又已經(jīng)達(dá)到-Xmx設(shè)置的***值。將會(huì)拋出OutOfMemoryError異常。例子如下:

  1. /**  
  2.   * VM Args: -Xmx5m  
  3.   */ 
  4.  public class Test {  
  5.        
  6.      public static void main(String[] args) {  
  7.          int count = 0;  
  8.          List<Object> list = new ArrayList<Object>();  
  9.          while (true) {  
  10.              list.add(new Object());  
  11.              System.out.println(++count);  
  12.          }  
  13.      }  
  14.    
  15.  } 

-Xmx為5m。其中的一次測(cè)試結(jié)果為,當(dāng)count的值累加到297868時(shí),發(fā)生如下異常:

  1. Exception in thread "main" java.lang.OutOfMemoryError: Java heap space  
  2.     at java.util.Arrays.copyOf(Arrays.java:2760)  
  3.     at java.util.Arrays.copyOf(Arrays.java:2734)  
  4.     at java.util.ArrayList.ensureCapacity(ArrayList.java:167)  
  5.     at java.util.ArrayList.add(ArrayList.java:351)  
  6.     at jvm.Test.main(Test.java:15

修改-Xmx為10m。其中的一次測(cè)試結(jié)果為,當(dāng)count的值累加到670205時(shí),發(fā)生OutOfMemoryError異常。隨著-Xmx參數(shù)值的增大,java堆中可以存儲(chǔ)的對(duì)象也越多。

方法區(qū)溢出

方法區(qū)用于存放java類型的相關(guān)信息,如類名、訪問(wèn)修飾符、常量池、字段描述、方法描述等。在類裝載器加載class文件到內(nèi)存的過(guò)程中,虛擬機(jī)會(huì)提取其中的類型信息,并將這些信息存儲(chǔ)到方法區(qū)。當(dāng)需要存儲(chǔ)類信息而方法區(qū)的內(nèi)存占用又已經(jīng)達(dá)到-XX:MaxPermSize設(shè)置的***值。將會(huì)拋出OutOfMemoryError異常。對(duì)于這種情況的測(cè)試,基本的思路是運(yùn)行時(shí)產(chǎn)生大量的類去填滿方法區(qū),直到溢出。這里需要借助CGLib直接操作字節(jié)碼運(yùn)行時(shí),生成了大量的動(dòng)態(tài)類。例子如下:

  1. /**  
  2.    * VM Args: -XX:MaxPermSize=50M  
  3.    */ 
  4.   public class Test {  
  5.         
  6.       public static void main(String[] args) {  
  7.           int count = 0;  
  8.           while (true) {  
  9.                Enhancer enhancer = new Enhancer();  
  10.                enhancer.setSuperclass(Test.class);  
  11.                enhancer.setUseCache(false);  
  12.                enhancer.setCallback(new MethodInterceptor() {  
  13.                   public Object intercept(Object obj, Method method, Object[] args,   
  14.                     MethodProxy proxy) throws Throwable {  
  15.                       return proxy.invoke(obj, args);  
  16.                   }  
  17.                });  
  18.                enhancer.create();  
  19.                System.out.println(++count);  
  20.           }  
  21.       }  
  22.     
  23.   } 

-XX:MaxPermSize為50m。其中的一次測(cè)試結(jié)果為,當(dāng)count的值累加到3953時(shí),發(fā)生如下異常:

  1. Caused by: java.lang.OutOfMemoryError: PermGen space  
  2.     at java.lang.ClassLoader.defineClass1(Native Method)  
  3.     at java.lang.ClassLoader.defineClassCond(ClassLoader.java:631)  
  4.     at java.lang.ClassLoader.defineClass(ClassLoader.java:615)  
  5.     ... 8 more 

修改-XX:MaxPermSize為100m。其中的一次測(cè)試結(jié)果為,當(dāng)count的值累加到8022時(shí),發(fā)生OutOfMemoryError異常。隨著-XX:MaxPermSize參數(shù)值的增大,java方法區(qū)中可以存儲(chǔ)的類型數(shù)據(jù)也越多。

 

相關(guān)的參考資料:

1.深入Java虛擬機(jī)(原書第2版)

2.深入理解Java虛擬機(jī):JVM高級(jí)特性與***實(shí)踐

3.互聯(lián)網(wǎng)相關(guān)的文章

原文鏈接:http://www.cnblogs.com/evan2012/archive/2012/05/12/2497086.html

【編輯推薦】

  1. JVM程序員的閱讀清單
  2. JVM加載過(guò)程及異常
  3. 當(dāng)下主流JVM語(yǔ)言一覽
責(zé)任編輯:林師授 來(lái)源: evan2012的博客
相關(guān)推薦

2020-08-10 17:49:25

JVM內(nèi)存溢出

2015-12-28 11:41:57

JVM內(nèi)存區(qū)域內(nèi)存溢出

2024-04-25 10:06:03

內(nèi)存泄漏

2023-03-03 12:37:50

JavaJVM內(nèi)存溢出

2010-09-26 15:53:25

JVM內(nèi)存溢出

2019-08-29 14:29:42

JVM內(nèi)存 Java

2018-12-04 10:54:20

JVM內(nèi)存模型

2010-09-26 16:04:48

JVM內(nèi)存溢出

2015-03-30 11:18:50

內(nèi)存管理Android

2024-09-09 09:41:03

內(nèi)存溢出golang開發(fā)者

2024-03-11 08:22:40

Java內(nèi)存泄漏

2013-08-02 10:06:36

Android內(nèi)存溢出

2018-12-20 10:17:35

JVM模型內(nèi)存溢出

2021-08-30 07:22:14

JVM OutOfMemory異常

2017-09-20 08:48:09

JVM內(nèi)存結(jié)構(gòu)

2021-03-06 10:25:19

內(nèi)存Java代碼

2023-08-01 08:20:42

JVM優(yōu)化虛擬機(jī)

2012-03-01 10:51:37

JavaJVM

2010-09-25 15:40:52

配置JVM內(nèi)存

2023-11-19 23:29:22

Heap DumpJava
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)