1

私は Java 同時実行の基本を学ぼうとしています。私はhttp://docs.oracle.com/javase/tutorial/essential/concurrency/deadlock.htmlまで Oracle Java チュートリアルから始めます。コード例にいくつかの行を追加するだけです。 、しかし、コードを実行すると、メインメソッドは返されません。alphonseThread と gastonThread の両方がそれぞれの ether によってデッドロックされていることはわかっていますが、メイン スレッドからそれらを中断できないのはなぜですか?

public class DeadLock {

     static class Freind {

          private final String name;
          public Freind(String name_) {
               this.name = name_;
          }

          public String getName() {
               return this.name;
          }

          public synchronized void bow(Freind bower) throws InterruptedException {
                System.out.format("%s : %s " + "has bowed to me!%n",
                      this.name, bower.getName());
                Thread.sleep(2000);
                bower.bowBack(this);

         }

         public synchronized void bowBack(Freind bower) {
               System.out.format("%s : %s " + "has bowed back to me!%n",
                this.name, bower.getName());
         }
     }

    @SuppressWarnings("SleepWhileInLoop")
    public static void main(String[] args) throws InterruptedException {
        // wait for 4 seconds before send a interuption signal
        int patient = 4000;
        int waitingUnit = 1000;
        long start;
        final Freind alphonse = new Freind("Alphonse");
        final Freind gaston = new Freind("Gaston");
        // the first thread 
        Thread alphonseThread = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    alphonse.bow(gaston);
                } catch (InterruptedException ex) {
                    System.out.println("Alphnso have been killed");
                }
             }
        });
        // the second thread
        Thread gastonThread = new Thread(new Runnable() {
            @Override
            public void run() {
                 try {
                     gaston.bow(alphonse);
                 } catch (InterruptedException ex) {
                     System.out.println("Gaston have been killed ");
                 }
             }
         });
         // start the tow threads                
         start = System.currentTimeMillis();
         alphonseThread.start();
         gastonThread.start();
         while (alphonseThread.isAlive() || gastonThread.isAlive()) {
             if (((System.currentTimeMillis() - start) < patient)
                     && (alphonseThread.isAlive() || gastonThread.isAlive())) {
                 System.out.println(System.currentTimeMillis() - start);
                 System.out.println("still waiting !!");
                 Thread.sleep(waitingUnit);
              } else {
                 if (alphonseThread.isAlive()) {
                     alphonseThread.interrupt();
                     alphonseThread.join();
                 }
                 if (gastonThread.isAlive()) {
                     gastonThread.interrupt();
                     gastonThread.join();
                }                     
             }
         }
         System.out.println("finnaly I kill all of them");
     }
 }
4

1 に答える 1

3

しかし、メインスレッドからそれらを中断できないのはなぜですか?

メインスレッドから (またはどこからでも) 中断できます。Thread.interrupt()割り込みたいスレッドを呼び出すだけです。

問題はinterrupt、Java プリミティブ ロック/ミューテックスを取得しようとしているスレッドのブロックを解除しないことです。実際、プリミティブ ロックでデッドロックされたスレッドを安全にブロック解除できるものはないと思います。

アプリケーションがオブジェクト ロックを伴うデッドロックから回復できるようにする場合は、おそらくLockオブジェクトとLock.lockInterruptibly()、スレッドが中断されたときにブロックを解除するメソッドを使用する必要があります。

于 2012-08-05T04:49:54.060 に答える