小编给大家分享一下Java异常之OutOfMemoryError怎么解决,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获、下面让我们一起去了解一下吧!
在Java虚拟机规范描述中,除了程序计数器外,虚拟机内存的其他几个运行区域都有发生伯父异常的可能。在这里,用代码验证各个运行时区域存储的内容并讨论该如何进行处理。
<强> Java堆溢出强>
Java堆用于存储对象实例,只要不断创建对象,并且保证GC根到对象之间有可达路径来避免垃圾回收机制清除这些对象,那么对象数量达到最大堆的容量限制之后就会产生内存溢出异常。
<强>异常再现强>
代码采用如下虚拟机参数:
-Xms20m -Xmx20m - xx: + HeapDumpOnOutOfMemoryError
这样的Java堆的大小将被限制为20 MB且不可拓展。通过参数<强> - xx: + HeapDumpOnOutOfMemoryError 强>可以让虚拟机在出现内存溢出异常时倾倒出当前的内存堆转储快照以便时候进行分析。
采用如下代码进行验证:
公共类HeapOOM { 静态类OOMObject { } 公共静态void main (String [] args) { List列表=new ArrayList (); 而(真){ 列表。添加(新OOMObject ()); } } }
运行结果:
. lang。OutOfMemoryError: Java堆空间 倾销java_pid3460堆。hprof…… 堆转储文件创建(28199779字节在0.237秒)
解决方法
Java堆内存的伯父异常是实际应用中常见的内存溢出异常情况,出现时往往会紧跟着提示“Java堆空间”。
要解决这个区域的异常,一般的手段是先通过内存映像分析工具,比如垫,确认到底是出现了内存泄漏还是内存溢出。
如果是内存泄漏,可以进一步通过工具查看泄漏对象到GC根的引用链,找到泄漏对象是通过怎样的途径和GC根相关联并导致垃圾收集器无法自动回收它们所占的空间。
如果不是内存泄漏,换而言之,内存中的对象确实还有必要存活着,那么就应当检查虚拟机的堆参数,与机器物理内存对比看是否还可以调大。从代码层面上看,是否存在某些对象生命周期过长,持有状态时间过长的情况,尝试减少程序运行期间的内存消耗。
虚拟机栈和本地方法栈溢出
由于在热点虚拟机中并不区分虚拟机栈或者本地方法栈,因此对于热点而言,虽然-Xoss参数存在,但是实际上是无效的,栈容量只由xss参数设定。
异常再现
在单线程下,代码采用如下的虚拟机参数:
- xss128k
使用该参数减小栈容量,使用如下代码复现异常:
公共类JavaVMStackSOF { 私人int stackLength=1; 公共空间stackLeak () { stackLength + +; stackLeak (); } 公共静态void main (String [] args)抛出Throwable { JavaVMStackSOF伯父=new JavaVMStackSOF (); 尝试{ oom.stackLeak (); }捕捉(Throwable e) { System.out.println(“堆栈长度:“+ oom.stackLength); 把e; } } }
解决方法
如果使用虚拟机默认参数,栈深度在大多数情况下(因为每个方法压入栈的帧大小并不是一样的,所以只能说在大多数情况下)达到1000 ~ 2000完全没有问题,对于正常的方法调用(包括递归),这个深度应该完全足够。
但是,如果是因为建立过多的线程导致内存溢出,在不能减少线程数或者更换64位虚拟机的情况下,就只能通过减少最大堆和减少栈容量来换取更多的线程。
<强>本机直接内存溢出强>
DirectMemory容量可以通过<强> - xx: MaxDirectMemorySize 强>指定,如果不指定,则默认与Java最大堆一样。
异常再现
使用以下虚拟机参数:
-Xmx20M - xx: MaxDirectMemorySize=10 m
使用以下代码重现异常:
公共类DirectMemoryOOM { 私有静态最终int _1MB=1024 * 1024; 公共静态void main (String [] args){抛出异常 场unsafeField=Unsafe.class.getDeclaredFields () [0]; unsafeField.setAccessible(真正的); 不安全不安全=(不安全)unsafeField.get(空); 而(真){ unsafe.allocateMemory (_1MB);//直接申请分配内存 } } }
解决方法
由DirectMemory导致的内存溢出,一个明显的特征就是在堆转储文件中不会看见明显的异常。