Java中的OutOfMemory种类和解决方法

 

1、OutOfMemory的四种情况

1) 永久区溢出    Exception in thread "main" java.lang.OutOfMemoryError: PermGen space  

这一部分用于存放Class和Meta的信息,Class在被 Load的时候被放入PermGen space区域(包括常量池: 静态变量),它和存放Instance的Heap区域不同,GC(Garbage Collection)不会在主程序运行期对PermGen space进行清理,所以如果你的APP会LOAD很多CLASS的话,就很可能出现PermGen space错误。

这种错误常见在web服务器对JSP进行pre compile的时候。

可以通过设置jvm启动参数来解决: -XX:MaxPermSize=256m

/** 
 * -XX:+HeapDumpOnOutOfMemoryError -XX:MaxPermSize=2m
 */
public class PermGenSpace {
    public static void main(String[] args) {
        List < String > list = new ArrayList < String > ();
        int i = 0;
        while (true) {
            list.add(String.valueOf(i++).intern());
        }
    }
}

2)堆溢出  java.lang.OutOfMemoryError: Java heap space

这部分用于存放类的实例。被缓存的实例(Cache)对象,大的map,list引用大的对象等等,都会保存于此。

堆内存会在jvm启动时自动设置,初始值 -Xms为物理内存的1/64,最大值-Xmx为1/4;可以通过参数-Xmn、-Xms、-Xmx设置,一般-Xms和-Xmx不超过80%,-Xmn为-Xmx的1/4;

/**  
 * -XX:+HeapDumpOnOutOfMemoryError -Xms20m -Xmx20m  -XX:MaxNewSize=10m 
 */
public class HeapSpace {
    public static void main(String[] args) {
        List < String > list = new ArrayList < String > ();
        int i = 0;
        while (true) {
            list.add(new String("" + i++));
        }
    }
}

3)java.lang.OutOfMemoryError: GC overhead limit exceeded

JDK6新增错误类型。当GC为释放很小空间占用大量时间时抛出。

/** 
 * -XX:+HeapDumpOnOutOfMemoryError -Xms20m -Xmx20m  -XX:MaxNewSize=256m 
 */
public class HeapSpace {
    public static void main(String[] args) {
        List < String > list = new ArrayList < String > ();
        int i = 0;
        while (true) {
            list.add(new String("" + i++));
        }
    }
}

可以添加JVM的启动参数来限制使用内存:-XX:-UseGCOverheadLimit

4)栈溢出错误,非规范的OutOfMemoryError     Exception in thread "main" java.lang.StackOverflowError

这部分用于存放局部变量、方法栈帧信息。栈帧太多,也就是函数调用层级过多时就会出现此异常,检查是否有死递归的情况。 

对应的启动参数为:-Xss ( JDK1.5以后默认是1M,之前是256K)

/** 
 * -XX:+HeapDumpOnOutOfMemoryError -Xss(默认1M) 
 */
public class StackOverflow {
    public int i = 0;
    public void stackLeak() {
        i++;
        stackLeak();
    }
    public static void main(String[] args) {
        StackOverflow sof = new StackOverflow();
        sof.stackLeak();
    }
}

实例,以下是1G内存环境下java jvm 的参数设置参考: 

JAVA_OPTS="-server -Xms800m -Xmx800m -XX:PermSize=64M -XX:MaxNewSize=256m -XX:MaxPermSize=128m -Djava.awt.headless=true 

2、dump排查方案

 jvm启动时添加参数-XX:+HeapDumpOnOutOfMemoryError,这样当内存溢出时,会生成dump文件:

 java.lang.OutOfMemoryError: Java heap space 

Dumping heap to java_pid5304.hprof ... 

Heap dump file created [85357895 bytes in 2.095 secs]

使用Memory Analyze分析,查看类使用情况;

3、分析工具

1)动态分析,实时 :  Jprofile

2)静态分析 ,需要dump文件:     Eclipse Memory Analyzer等

原文:http://wade6.iteye.com/blog/1978147


赞(52) 打赏
未经允许不得转载:优客志 » JAVA开发
分享到:

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

微信扫一扫打赏