OutOfMemory的三种情况

摘自kakaluyi的<如何定位OutOfMemory的根本原因>,有部分修改

java的OutOfMemory有关Exception和可能出现的方式:

A Exception in thread "main" java.lang.OutOfMemoryError: PermGen space
Java代码 复制代码  收藏代码
  1. public static void main(String[] args) {   
  2.     //使用List保持着常量池引用,压制Full GC回收常量池行为   
  3.     List<String> list = new ArrayList<String>();   
  4.     // 10M的PermSize在integer范围内足够产生OOM了   
  5.     int i = 0;   
  6.     while (true) {   
  7.         list.add(String.valueOf(i++).intern());   
  8.     }   
  9. }  
public static void main(String[] args) {
	//使用List保持着常量池引用,压制Full GC回收常量池行为
	List<String> list = new ArrayList<String>();
	// 10M的PermSize在integer范围内足够产生OOM了
	int i = 0;
	while (true) {
		list.add(String.valueOf(i++).intern());
	}
}

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

B java.lang.OutOfMemoryError: Java heap space
Java代码 复制代码  收藏代码
  1. public static void main(String[] args) {   
  2.     //使用List保持着常量池引用,压制Full GC回收常量池行为   
  3.     List<String> list = new ArrayList<String>();   
  4.     // 10M的PermSize在integer范围内足够产生OOM了   
  5.     int i = 0;   
  6.     while (true) {   
  7.         list.add(new String(i++));   
  8.     }   
  9. }  
public static void main(String[] args) {
	//使用List保持着常量池引用,压制Full GC回收常量池行为
	List<String> list = new ArrayList<String>();
	// 10M的PermSize在integer范围内足够产生OOM了
	int i = 0;
	while (true) {
		list.add(new String(i++));
	}
}

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

C Exception in thread "main" java.lang.StackOverflowError
Java代码 复制代码  收藏代码
  1. public class JavaVMStackSOF {   
  2.     private int stackLength = 1;   
  3.     public void stackLeak() {   
  4.         stackLength++;   
  5.         stackLeak();   
  6.     }   
  7.     public static void main(String[] args) throws Throwable {   
  8.         JavaVMStackSOF oom = new JavaVMStackSOF();   
  9.         try {   
  10.             oom.stackLeak();   
  11.         } catch (Throwable e) {   
  12.             System.out.println("stack length:" + oom.stackLength);   
  13.             throw e;   
  14.         }   
  15.     }   
  16. }  
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;
		}
	}
}

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

今天碰到个内存溢出的问题,解决过程如下:
    • 首先启动时添加参数-XX:+HeapDumpOnOutOfMemoryError,这样当内存溢出时:        java.lang.OutOfMemoryError: Java heap space

    •        Dumping heap to java_pid5304.hprof ...

  •        Heap dump file created [85357895 bytes in 2.095 secs]
  • 会在eclipse目录下生产内存文件;
  • 使用Memory Analyze分析,查看类使用情况,发现是由Map过大引起;
  • 查看代码中使用到Map的段,分析,解决问题;


问题over。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值