gpt4 book ai didi

java - 如何获取 StackOverflowError 的完整堆栈

转载 作者:IT老高 更新时间:2023-10-28 21:06:01 24 4
gpt4 key购买 nike

观察 StackOverflowError 时如何检索完整的调用堆栈?

考虑这个简单的例子:

public class Overflow {

public Overflow() {
new Overflow();
}
public static void a() {
new Overflow();
}
public static void main(String[] argv) {
a();
}
}

现在报错是:

Exception in thread "main" java.lang.StackOverflowError
at Overflow.<init>(Overflow.java:11)
[last line repeated many times]

但我在堆栈跟踪中看不到 maina 方法。我的猜测是这是因为溢出,堆栈上的最新条目替换了最旧的条目(?)。

现在,如何在输出中获取 amain 堆栈条目?

背景是我得到一个 StackOverflowError (但这不是无限递归,因为它不会在增加堆栈大小时发生)并且很难在代码中发现问题。我只从 java.util.regex.Pattern 得到多行,但没有得到代码调用的信息。应用程序过于复杂,无法在每次调用 Patterns 时设置断点。

最佳答案

JVM 人为限制了 1024 个条目,您可以在异常或错误的堆栈跟踪中拥有这些条目,可能是为了在发生异常或错误时节省内存(因为 VM 必须分配内存来存储堆栈跟踪)。

幸运的是,有一个标志可以增加这个限制。只需使用以下参数运行您的程序:

-XX:MaxJavaStackTraceDepth=1000000

这将打印多达 100 万个堆栈跟踪条目,这应该绰绰有余。也可以将此值设置为 0 以将条目数设置为无限。

This list of non-standard JVM options提供更多细节:

Max. no. of lines in the stack trace for Java exceptions (0 means all). With Java > 1.6, value 0 really means 0. value -1 or any negative number must be specified to print all the stack (tested with 1.6.0_22, 1.7.0 on Windows). With Java <= 1.5, value 0 means everything, JVM chokes on negative number (tested with 1.5.0_22 on Windows).

使用此标志运行问题示例会得到以下结果:

Exception in thread "main" java.lang.StackOverflowError
at Overflow.<init>(Overflow.java:3)
at Overflow.<init>(Overflow.java:4)
at Overflow.<init>(Overflow.java:4)
at Overflow.<init>(Overflow.java:4)
(more than ten thousand lines later:)
at Overflow.<init>(Overflow.java:4)
at Overflow.<init>(Overflow.java:4)
at Overflow.a(Overflow.java:7)
at Overflow.main(Overflow.java:10)

这样,即使实际堆栈跟踪的长度超过 1024 行,您也可以找到引发错误的代码的原始调用者。

如果您不能使用该选项,还有另一种方法,如果您在这样的递归函数中,并且可以修改它。如果添加以下try-catch:

public Overflow() {
try {
new Overflow();
}
catch(StackOverflowError e) {
StackTraceElement[] stackTrace = e.getStackTrace();
// if the stack trace length is at the limit , throw a new StackOverflowError, which will have one entry less in it.
if (stackTrace.length == 1024) {
throw new StackOverflowError();
}
throw e; // if it is small enough, just rethrow it.
}
}

本质上,这将创建并抛出一个新的 StackOverflowError,丢弃最后一个条目,因为与前一个相比,每个条目都将被发送一个级别(这可能需要几秒钟,因为所有这些必须创建错误)。当堆栈跟踪将减少到 1023 个元素时,它会简单地重新抛出。

最终这将打印堆栈跟踪底部的 1023 行,这不是完整的堆栈跟踪,但可能是其中最有用的部分。

关于java - 如何获取 StackOverflowError 的完整堆栈,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5165753/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com