eclipse编码实现内存溢出异常

1.堆内存溢出
代码如下:

package OOM;
import java.util.ArrayList;
import java.util.List;
public class HeapOOM {
	static class OOMOBject {
	}
	public static void main(String[] args) {
		List<OOMOBject> list = new ArrayList<>();
		while (true) {
			list.add(new OOMOBject());
			System.out.println(list.size());
		}
	}
}

运行参数如下:设置堆的大小为20M,能够很快看到异常信息。

-verbose:ge -Xms20M -Xmx20M -XX:+PrintGCDetails -XX:SurvivorRatio=9
异常信息如下:
java.lang.OutOfMemoryError: Java heap space

2.虚拟机栈和本地方法栈溢出
hotSpot虚拟机不区分虚拟机栈和本地方法栈,栈容量只由-Xss参数设定。如果线程请求的栈深度大于虚拟机所允许的最大深度,将跑出stackOverflowError,如何虚拟机在扩展栈是无法申请到足够的内存空间,将抛出OutOfMemoryError异常。

package OOM;
public class JavaVMStackSOF {
	private int stackLength =1;
	public void stackLeak() {
		stackLength++;
		System.out.println("stack length:"+stackLength);
		stackLeak();
	}

	public static void main(String[] args)  {
		JavaVMStackSOF oom=new JavaVMStackSOF();
		oom.stackLeak();
	}
}
异常如下:
stack length:5838
Exception in thread "main" java.lang.StackOverflowError
package OOM;
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) {
		JavaVMStackOOM oom=new JavaVMStackOOM();
		oom.stackLeakByThread();
	}
}
这段代码运行后系统假死

3.运行时常量池溢出
如果要向运行时常量池添加内容,最简单的做法就是使用String.intern()这个Native方法。改方法的作用是:如果池中已经包含一个等于此String对象的字符串,则返回代表池中这个字符串String对象;否则,将此string对象包含的字符串添加常量中,并且返回此string对象的引用。由于常量池分配在方法区内,我们可以通过-XX:permSize和-XX:MaxPermSize限制方法区的大小,从而间接限制其中常量池的容量。

package OOM;
import java.util.ArrayList;
import java.util.List;
public class RuntimeConstantPoolOOM {
	public static void main(String[] args) {
		List<String> list=new ArrayList<String>();
		int i=0;
		while(true) {
			list.add(String.valueOf(i++).intern());
		}
	}
}
这段代码运行后系统假死

4.方法区溢出
通过cglib动态生成class可以载入内存。运行参数-verbose:gc -XX:PermSize=1M -XX:MaxPermSize=1M

package OOM;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class JavaMethodAreaOOM {
	static class OOMObject{
	}
	public static void main(String[] args) {
		while(true) {
			Enhancer enhancer=new Enhancer();
			enhancer.setSuperclass(OOMObject.class);
			enhancer.setUseCache(false);
			enhancer.setCallback(new MethodInterceptor() {
				@Override
				public Object intercept(Object arg0, Method arg1, Object[] arg2, MethodProxy arg3) throws Throwable {
					return arg3.invokeSuper(arg0, arg2);
				}
			});
			enhancer.create();
			System.out.println(enhancer.hashCode());
		}
	}
}
一直运行良好,未发现异常

5.本机直接内存溢出
直接运行内在并不是虚拟机运行时数据区的一部分,也不是java虚拟机规范中定义的内存区域,在jdk1.4中新加入NIO类,引入 了一种基于通道与缓冲区的io方式,它可以使用native函数直接分配堆外内存,然后通过一个存储在java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。本机直接内在的分配不会受到java堆大小的限制,受到本机总内在大小限制,配置虚拟机参数时,不要忽略直接内存。防止outofMemory异常。directMemory容量可以通过-XX:MaxDirectMemorySize指定,如果不指定则默认与Java堆的最大值一样。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值