1

次のコードを検討してください:-

public class UsingWait1{
public static void main(String... aaa){
    CalculateSeries r = new CalculateSeries();
    Thread t = new Thread(r);
    t.start();
    synchronized(r){
        try{
            r.wait();   //Here I am waiting on an object which is Runnable. So from its run method, it can notify me (from inside a synchronized block).
        } catch (InterruptedException e) {
            System.out.println("Interrupted");
        }

    }
    System.out.println(r.total);
    try{
        Thread.sleep(1);
    } catch (InterruptedException e){
        System.out.println("Interrupted");
    }
    System.out.println(r.total);

}
}

class CalculateSeries implements Runnable{
int total;
public void run(){
synchronized(this){
    for(int i = 1; i <= 10000; i++){
        total += i;
    }
    notify();   // Line 1 .. Notify Exactly one of all the threads waiting on this instance of the class to wake up
}
}
}

ここでは、実行可能なCalculateSeriesを待っています。したがって、CalculateSeriesのrun()メソッドから待機中のスレッドに通知できます。

しかし、ここで、実行可能ではないオブジェクトを待機している次のコードについて考えてみます。

public class WaitNotOnThread{
public static void main(String... aaa){
    NotRunnable nr = new NotRunnable();
    IAmRunnable r = new IAmRunnable(nr);
    new Thread(r).start();
    synchronized(nr){
        try{
            nr.wait();
        } catch(InterruptedException e){
            System.out.println("Wait interrupted");             
        }

        System.out.println("After being notified within synchronized");
    }
    System.out.println("After synchronized");
}
}

class IAmRunnable implements Runnable{
NotRunnable nr;
IAmRunnable(NotRunnable nr){
    this.nr = nr;
}
public void run(){
    synchronized(nr){
        try{
            Thread.sleep(1000);
        } catch(InterruptedException e){
            System.out.println("Sleeping Interrupted :( ");
        }
        notify();                  // Line 2
    }
}
}

class NotRunnable{
}

ここでは、2行目でIllegalMonitorStateExceptionが発生します。wait()とnotify()の両方を呼び出している間、オブジェクトの同じインスタンス(Runnableではない)を待機しています。では、何が問題なのでしょうか。

誰かが実行可能ではないオブジェクトを待つことが役立ついくつかのシナリオを与えることもできますか?

4

1 に答える 1

4

Runnableで待機する必要はありません。そのため、notify()はRunnable ではなくObjectにあります。ビジーウェイトを避けたいすべての場合に役立つと思います。

問題は、synchronized()がnrにあり、notifyが別のオブジェクトで呼び出されていることのようです。また、最終変数で同期する必要があります。

class IAmRunnable implements Runnable {
     final NotRunnable nr;

    IAmRunnable( final NotRunnable nr) {
        this.nr = nr;
    }

    public void run() {
        synchronized (nr) {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                System.out.println("Sleeping Interrupted :( ");
            }
            nr.notify();                  // Line 2
        }
    }
}
于 2012-01-25T06:00:32.963 に答える