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 のドキュメントのRuntime.exitには、スレッド要件に関するドキュメントはありません。

Linux でのみ AWT ツリー ロックを取得する必要があるという驚くべきケースを他にも見つけましたが、これはうまくいきます。これはバグですか、それともドキュメントで何かを見逃したのですか?

4

2 に答える 2

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 に答える
1

それによって異なりますが、runHooksメソッドは登録されたフックスレッドを開始しRuntime.addShutdownHook、それらが終了するのを待ちます。フック スレッドのいずれかが、AWT イベント スレッドも必要とするリソースをロックしている場合、デッド ロックが発生する可能性があります。

AWT イベント スレッドで System.exit を呼び出す必要がある場合は、次のような別のスレッドで呼び出すことをお勧めします。

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