JVM不会在OnOutOfMemoryError上被杀死

时间:2018-12-24 09:05:05

标签: java jvm out-of-memory

我正在使用JDK 1.8.0_u66选项尚不可用的-XX:+CrashOnOutOfMemoryError。所以我最终设置了-XX:OnOutOfMemoryError="kill -9 %p"

好的,我编写了以下简单的应用程序:

public static void main(String[] args) throws Exception{
    Lock lock = new ReentrantLock();
    Condition condition = lock.newCondition();

    while (true) {
        new Thread(() -> {
            lock.lock();
            try {
                while (true) {
                    try{
                        condition.await();
                    } catch (Exception e){}
                }
            } finally {
                lock.unlock();
            }
        }).start();
    }
}

应用程序失败,并显示错误

Exception in thread "main" java.lang.OutOfMemoryError: 
unable to create new native thread

很快,但是由于设置了选项-XX:OnOutOfMemoryError="kill -9 %p",它没有像我预期的那样被杀死。

我猜想杀死JVM是在单独的Thread中完成的,但是由于已经达到限制,因此无法创建一个。因此,应用程序会默默地保持工作在不一致状态。

1 个答案:

答案 0 :(得分:3)

您最有可能受到JDK-8155004的影响,这使得-XX:OnOutOfMemoryError和相关选项在创建线程OutOfMemoryError时无法运行。

OutOfMemoryError是由堆内存不足引起的时,您当前的方法效果很好。您可以使用以下代码和-XX:OnOutOfMemoryError="kill -9 %p"选项进行测试。它退出JVM 1.8.0_191(离您的版本最近):

public static void main() {
  Thread t = new Thread(() -> {
    try {
      TimeUnit.HOURS.sleep(30);
    } catch (InterruptedException e) {}
  });
  t.start(); // to prevent natural JVM exit when main dies

  ArrayList<Long[]> retain = new ArrayList<>();
  while (true) {
    Long[] arr = new Long[Integer.MAX_VALUE];
    retain.add(arr);
  }
}