为什么Java forkbomb不会导致StackOverflowError?

时间:2013-01-15 14:21:09

标签: java stack-overflow

我是编程新手,我正在尝试自学一下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();
   }
}

3 个答案:

答案 0 :(得分:9)

每个线程都有自己的堆栈。创建新进程时,还会获得新线程和新堆栈。填充单个堆栈时会发生堆栈溢出错误,但是您创建了许多不同的堆栈,其中没有一个堆栈已满。

在第二个示例中,您只有一个线程,并且您有一个递归调用。每次调用方法时,某些信息都会临时存储在堆栈中,直到方法返回为止。因为您的方法永远不会返回,所以堆栈将被消耗,直到您遇到堆栈溢出异常。

你甚至不需要while循环。这也会产生堆栈溢出异常:

public WhileBomb()
{
   new WhileBomb();
}

答案 1 :(得分:2)

第一个例子是没有向堆栈添加任何东西,它正在创建全新的堆栈。但是,第二个例子在这里添加了堆栈:

public WhileBomb()
{
    while (true)
    {
        new WhileBomb();
    }
}

你在这里看到的是无限递归。创建WhileBomb对象涉及创建WhileBomb对象。这一切都发生在同一个程序中的同一个线程中(不会分离出新程序),因此它位于同一个堆栈中。在线程中每次创建新的WhileBomb都会将自己的另一个实例添加到堆栈中。

答案 2 :(得分:1)

堆栈溢出是由使用的堆栈空间多于可用堆栈空间引起的。一个叉炸弹通过产生大量新进程来减慢PC的速度,每个进程都会产生更多的进程(等等)。

fork炸弹不会导致堆栈溢出的原因是每个进程都有自己的堆栈,因此你实际上并没有在任何进程中使用太多堆栈空间,只创建了许多进程,每个进程都有自己的堆栈空间(相对较小)堆叠。