我是编程新手,我正在尝试自学一下StackOverflow是由什么造成的。我玩循环并导致错误,但我测试的forkbomb代码不会导致错误。这是为什么?
public class ForkBomb
{
public static void main(String[] args) throws java.io.IOException
{
while(true)
{
Runtime.getRuntime().exec(new String[]{"java", "-cp", System.getProperty("java.class.path"), "ForkBomb"});
}
}
}
这不会导致堆栈溢出。为什么?然而这导致一个:
public class WhileBomb
{
public WhileBomb()
{
while (true)
{
new WhileBomb();
}
}
public static void main(String[] args) throws java.io.IOException
{
WhileBomb goodBye = new WhileBomb();
}
}
答案 0 :(得分:9)
每个线程都有自己的堆栈。创建新进程时,还会获得新线程和新堆栈。填充单个堆栈时会发生堆栈溢出错误,但是您创建了许多不同的堆栈,其中没有一个堆栈已满。
在第二个示例中,您只有一个线程,并且您有一个递归调用。每次调用方法时,某些信息都会临时存储在堆栈中,直到方法返回为止。因为您的方法永远不会返回,所以堆栈将被消耗,直到您遇到堆栈溢出异常。
你甚至不需要while循环。这也会产生堆栈溢出异常:
public WhileBomb()
{
new WhileBomb();
}
答案 1 :(得分:2)
第一个例子是没有向堆栈添加任何东西,它正在创建全新的堆栈。但是,第二个例子在这里添加了堆栈:
public WhileBomb()
{
while (true)
{
new WhileBomb();
}
}
你在这里看到的是无限递归。创建WhileBomb
对象涉及创建WhileBomb
对象。这一切都发生在同一个程序中的同一个线程中(不会分离出新程序),因此它位于同一个堆栈中。在线程中每次创建新的WhileBomb
都会将自己的另一个实例添加到堆栈中。
答案 2 :(得分:1)
堆栈溢出是由使用的堆栈空间多于可用堆栈空间引起的。一个叉炸弹通过产生大量新进程来减慢PC的速度,每个进程都会产生更多的进程(等等)。
fork炸弹不会导致堆栈溢出的原因是每个进程都有自己的堆栈,因此你实际上并没有在任何进程中使用太多堆栈空间,只创建了许多进程,每个进程都有自己的堆栈空间(相对较小)堆叠。