Java内存溢出异常(上)

这是一个常见的OOM异常。对于这种异常,在打印异常信息时通常会引发异常的原因,如图中所示的“Java堆空间”。当然,仅此信息还不足以确定内存容量设置是否很小或是否存在内存泄漏(有关内存泄漏的知识将在后面的文章中介绍)。因此,我们需要通过添加-xx:+heapDumpOnAutofMemoryError参数等其他方式进一步查明问题的根本原因,使dump在虚拟机发生内存溢出异常时,取当前内存堆转储快照,并用相关工具进行分析。这类知识在本文中不会解释得太多,但将在下面的文章中逐一介绍。

虚拟机栈和本地方法栈溢出

为什么要一起讨论虚拟机堆栈和本地方法堆栈的溢出,因为热点虚拟机中的虚拟机堆栈和本地方法堆栈没有区别。对于Hotspot,虽然-xoss参数用于设置本地方法堆栈的大小,但它实际上是无效的,堆栈的容量仅由-xss参数设置。

在Java虚拟机规范中,针对虚拟机栈和本地方法栈描述了两种异常:

  1. 如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常。

  2. 如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常。

这种分类不是很清楚,因为太少的内存或太多的堆栈空间会导致堆栈空间无法继续分配。

堆栈溢出错误是一个简单的条件。堆栈溢出错误是以下简单代码中的堆栈溢出:堆栈溢出错误、堆栈溢出错误、堆栈溢出错误、堆栈溢出错误、堆栈溢出错误、堆栈溢出错误、堆栈溢出错误

public class JavaVMStackSOF {

private int stackLength = 1;

public void stackLeak() {

stackLength++;

stackLeak();

}

public static void main(String[] args) {

JavaVMStackSOF javaVMStackSOF = new JavaVMStackSOF();

try {

javaVMStackSOF.stackLeak();

} catch (Throwable e) {

System.out.println(“Stack length:” + javaVMStackSOF.stackLength);

throw e;

}

}

}

运行结果如下:

Stack length:18663

Exception in thread “main” java.lang.StackOverflowError

at cn.bdqfork.jvm.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:12)

at cn.bdqfork.jvm.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:12)

at cn.bdqfork.jvm.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:12)

at cn.bdqfork.jvm.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:12)

对于上述结果,不同计算机的堆栈长度大小是不确定的。从输出异常信息来看,这是因为stackleak方法具有太多的递归调用层。在大多数情况下,在虚拟机的默认参数下,堆栈深度足够。

OutofMemoryError异常相对比较难发生,通常在多线程环境中。创建线程时,虚拟机会会将私有堆栈空间分配给相应的线程,该线程的大小可以用-xss参数设置。通过不断地创建新进程,可以生成内存溢出异常。

原因是当一个进程运行时,操作系统分配给该进程的内存是有限的。Java堆和方法区域占大多数内存,忽略程序计数器占用的一小部分内存,而不计算虚拟机本身占用的内存,其余部分由虚拟机栈和本地方法栈占据。因此,当创建的线程数达到一定水平时,虚拟机堆栈和本地方法堆栈占用的空间会使进程的内存空间不足,从而抛出内存溢出异常。

这部分的测试代码如下:

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();

}

}

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java内存溢出异常(OutOfMemoryError)不能被捕获。这是因为当内存溢出时,Java虚拟机(JVM)无法提供足够的内存来分配新的对象,因此无法继续运行。在这种情况下,JVM不会将异常传递给catch块,而是直接终止程序。因此,Java内存溢出异常是无法被捕获的。 ### 回答2: Java内存溢出异常(OutOfMemoryError)是一种严重的错误,它表示Java虚拟机无法为应用程序分配足够的内存空间。这种异常无法被普通的异常处理机制捕获和处理。 当Java应用程序运行时,Java虚拟机会将内存分为堆(Heap)和栈(Stack)两个部分。堆用于存储对象实例,而栈用于存储方法调用和局部变量。当应用程序试图创建一个新的对象实例或调用方法时,Java虚拟机会在堆或栈上分配相应的内存空间。 如果应用程序需要创建的对象过多,或者递归调用的层次太深导致栈空间耗尽,就会发生内存溢出异常。此时,Java虚拟机无法分配更多的内存空间,导致应用程序无法运行。 由于内存溢出异常不属于普通的异常类型(Throwable),因此无法被try-catch块捕获。尽管可以使用try-catch语句来捕获其他异常,但内存溢出异常会导致应用程序直接崩溃,无法再执行进一步的操作。在发生内存溢出异常时,通常会打印相关的错误信息,并且无法通过捕获和处理该异常来修复应用程序。 为了解决内存溢出异常,通常需要对应用程序进行优化,如减少对象的创建和引用、释放无用的对象、调整堆栈的大小等措施,以提高内存的使用效率和性能。 ### 回答3: Java中的内存溢出异常(OutOfMemoryError)一般情况下是无法被捕获的。内存溢出是指程序在申请内存时,无法获取到所需的内存空间而导致的异常。这种异常通常发生在堆内存空间不足以满足程序的需求时,例如创建过多的对象或者加载过大的数据。 由于内存溢出异常涉及到底层内存管理,是由Java虚拟机抛出的致命错误,无法通过传统的方式捕获和处理。一旦内存溢出发生,Java虚拟机将无法继续执行程序,直接导致程序崩溃。 但是,我们可以通过一些手段来预防和避免内存溢出异常的发生。例如,合理管理和释放对象的内存,避免创建过多且无用的对象;增加JVM堆内存的大小,确保程序有足够的内存空间;使用较新的JDK版本,其中对内存管理的优化可能有助于减少内存溢出的风险。 此外,还可以通过监控和分析程序的内存使用情况来及时发现潜在的内存溢出问题,并进行优化和调整。例如,使用一些内存分析工具,如VisualVM、Eclipse Memory Analyzer等,来检查程序的内存占用情况和对象泄漏情况,以及查找内存使用过多的地方。 总之,虽然无法直接捕获Java内存溢出异常,但通过合理的内存管理和优化,可以提高程序的稳定性和性能,减少内存溢出的风险。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值