考虑以下生成 StackOverflowError 的代码。
public class SimpleProgram {
static SimpleProgram s = new SimpleProgram();
public static void main(String[] args) {
s.f(0);
}
void f(int i) {
System.out.println("f :" + (++i));
g(i);
}
void g(int i) {
System.out.println("g :" + (++i));
f(i);
}
}
它只是打印从 1 到 n 的数字(我执行时 f() 的 '5417' )。之后它抛出 StackOverflowError 并终止。现在考虑第二个程序。
public class SimpleProgram{
static SimplePrograms = new SimpleProgram();
public static void main(String[] args) {
s.f(0);
}
void f(int i) {
try {
System.out.println("f :" + (++i));
g(i);
} catch (StackOverflowError e) {
System.out.println("f :" + (++i));
g(i);
}
}
void g(int i) {
try {
System.out.println("g :" + (++i));
f(i);
} catch (StackOverflowError e) {
System.out.println("g :" + (++i));
f(i);
}
}
}
现在它显示了一些奇怪的行为。程序没有按预期终止,但显示的值会重复(比如 f :4107 g :4108 到 f :4120 并再次返回 f :4107)。
我的问题是,为什么会发生这种情况?我认为像 StackOverflowError 这样的错误意味着当前线程的堆栈已满,因此无法恢复。程序必须强行停止它的执行,即不调用 next 函数,但没有发生。
JVM 线程堆栈可以按需增加其大小吗?
最佳答案
您的异常处理程序进行了更多的方法调用(对 System.out.println()
和 f()
或 g()
)。这些调用可能会抛出一个新的 StackOverflowError
,它可能会在调用堆栈的几个级别下被捕获,因此您会看到数字无限下降和备份(因为异常下降到调用堆栈中的方法,一些空间是在调用堆栈中释放,因此有空间进行额外的调用)。
这是我得到的部分输出(为了更清楚,我在 catch 块内的 println
语句中添加了“catch”):
第一个 StackOverflowError
发生在 System.out.println(String)
尝试调用 println()
时,这就是 g :9662catch g :9663
中不显示新行的原因。出于某种原因,在第一个 StackOverflowError
之后,还有大约 50 个额外调用的空间,直到我们得到另一个 StackOverflowError
,之后错误会更周期性地出现。
... no eception till this point...
g :9660
f :9661
g :9662catch g :9663
f :9664
g :9665
f :9666
g :9667
f :9668
g :9669
f :9670
g :9671
f :9672
g :9673
f :9674
g :9675
f :9676
g :9677
f :9678
g :9679
f :9680
g :9681
f :9682
g :9683
f :9684
g :9685
f :9686
g :9687
f :9688
g :9689
f :9690
g :9691
f :9692
g :9693
f :9694
g :9695
f :9696
g :9697
f :9698
g :9699
f :9700
g :9701
f :9702
g :9703catch g :9704catch f :9703
g :9704catch g :9705catch g :9702
f :9703
g :9704catch g :9705catch f :9704
g :9705catch g :9706catch f :9701
g :9702
f :9703
g :9704catch g :9705catch f :9704
g :9705catch g :9706catch g :9703
f :9704
g :9705catch g :9706catch f :9705
g :9706catch g :9707catch g :9700
f :9701
g :9702
f :9703
g :9704catch g :9705catch f :9704
g :9705catch g :9706catch g :9703
f :9704
g :9705catch g :9706catch f :9705
g :9706catch g :9707catch f :9702
g :9703
f :9704
g :9705catch g :9706catch f :9705
g :9706catch g :9707catch g :9704
f :9705
....
关于java - Java 中是否可以进行错误恢复?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/29430490/