6

我刚刚从 Oracle JDK 1.6 切换到 Open JDK 1.7.0_03,并且在退出时遇到了一个相当显着的死锁:

java.lang.Thread.State: WAITING (on object monitor)
 at java.lang.Object.wait(Native Method)
 at java.lang.Thread.join(Thread.java:1258)
 - locked <0x8608dda0> (a sun.awt.X11.XToolkit$1$1)
 at java.lang.Thread.join(Thread.java:1332)
 at java.lang.ApplicationShutdownHooks.runHooks(ApplicationShutdownHooks.java:106)
 at java.lang.ApplicationShutdownHooks$1.run(ApplicationShutdownHooks.java:46)
 at java.lang.Shutdown.runHooks(Shutdown.java:123)
 at java.lang.Shutdown.sequence(Shutdown.java:167)
 at java.lang.Shutdown.exit(Shutdown.java:212)
 - locked <0x8603df28> (a java.lang.Class for java.lang.Shutdown)
 at java.lang.Runtime.exit(Runtime.java:107)
 at java.lang.System.exit(System.java:960)

看来您必须从 AWT 事件队列中调用 System.exit。这是真的吗?Sun docs Runtime.exit中没有关于线程要求的文档

我遇到了其他令人惊讶的情况,即仅在 Linux 上才需要获取 AWT 树锁,但这个情况很重要。这是一个错误,还是我只是错过了文档中的某些内容?

4

2 回答 2

1

这取决于,该runHooks方法将启动通过注册的任何钩子线程Runtime.addShutdownHook并等待它们完成。如果您的任何钩子线程正在锁定 AWT 事件线程也需要的某些资源,它们可能会导致死锁。

如果您必须在 AWT 事件线程中调用 System.exit,我建议您在另一个线程中调用它,例如:

 new Thread(){
            public void run() {
                System.exit(0);
            }
   }.start();
于 2013-04-04T14:45:56.217 回答
1

如果不了解应用程序正在做什么(理想情况下,这将采用SSCCE的形式),就不可能说这是否是运行时中的错误。

例如,下面演示了一个类似的死锁,涉及System.exit(). 但是,这显然是应用程序中的错误,而不是System.exit()

public class OhNo {

    final static Object lock = new Object();

    public static void main(String[] args) {
        new Thread(new Runnable() {
            public void run() {
                synchronized (lock) {
                    for (;;) {
                    }
                }
            }
        }).start();
        Runtime.getRuntime().addShutdownHook(new Thread(new Runnable() {
            public void run() {
                synchronized (lock) {
                    System.out.println("in shutdown hook");
                }
            }
        }));
        System.out.println("about to call System.exit()");
        System.exit(0);
    }
}
于 2013-04-04T14:54:59.833 回答