我刚刚从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文档Runtime.exit
中没有关于线程要求的文档我遇到了其他令人惊讶的情况,只有在Linux上才需要获取AWT树锁,但是这个需要蛋糕。这是一个错误,还是我错过了文档中的内容?
答案 0 :(得分:1)
这取决于,runHooks
方法将启动通过Runtime.addShutdownHook
注册的任何钩子线程并等待它们完成。如果你的任何钩子线程都锁定了AWT事件线程所需的一些资源,它们可能会导致死锁。
如果你必须在你的AWT事件线程中调用System.exit,我建议你在另一个线程中调用它:
new Thread(){
public void run() {
System.exit(0);
}
}.start();
答案 1 :(得分: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);
}
}