程序暂停:wait() 和 notify()

Program gets halted: wait() and notify()

我正在努力实现这一点:创建了两个不同的线程,一个打印奇数,一个打印偶数。一旦一个线程打印了一个数字,它就必须等待另一个线程等等,一个接一个。

为了实现这一点,我将同步块与 wait() 和 notify() 一起使用。

我正在创建一个 class 其对象将用于传递给两个线程中的同步块。

代码如下:

--> 这是将传递给同步块的已用对象。

package com.vipin.multithread.variousdemos;

    public class SyncObject {

        public SyncObject () {  

        }
}

奇数线程:

package com.vipin.multithread.variousdemos;

public class OddThread implements Runnable {

private Thread t;
int index=0;
SyncObject so=null;

int odd_nums[] = {1,3,5,7,9};

public OddThread(SyncObject so) {
    t = new Thread(this,"Odd Thread");
    this.so = so;
    t.start();
}

public Thread getThreadInstance() {
    return t;
}

@Override
public void run() {
    while (true) {
        synchronized(so) {
            System.out.println("Odd num is --->" + odd_nums[index]);
            try {
                so.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            index++;
            so.notify();
            if(index>=5) {
                return;
            }
        }
    }
}
}

偶线程:更新

package com.vipin.multithread.variousdemos;

public class EvenThread implements Runnable {

private Thread t;
int index=0;
SyncObject so=null;

int even_nums[] = {2,4,6,8,10};

public EvenThread(SyncObject so) {
    t = new Thread(this, "Even thread");
    this.so = so;
    t.start();
}

public Thread getThreadInstance() {
    return t;
}

@Override
public void run() {
    while(true) {   
        synchronized(so) {
            System.out.println("Even num is --->" + even_nums[index]);
            so.notify(); <-- Here we are notifying.
            try {
                so.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            index++;
            //so.notify(); <-- commented out.
            if(index>=5) {
                break;
            }
        }
    }
}
}

主要应用:

package com.vipin.multithread.variousdemos;

public class EvenOddDemo {

public static void main(String[] args) throws InterruptedException {
    SyncObject so = new SyncObject();

    OddThread ot = new OddThread(so);
    EvenThread et = new EvenThread(so);

    System.out.println("\nIn main thread");

    Thread.sleep(1000000000);

    System.out.println("Exiting main thread...");
}
}

---> 如代码所示,我正在创建两个线程来打印偶数和奇数。我正在使用同步块,并传递 ==> SyncObject 类型的对象。

我将 SyncObject 作为参数传递给 main 中的这些不同线程。

但是,这个程序会停止,即只执行第一个语句,然后永远等待:

这是输出:

奇数是--->1

在主线程中 偶数是 --->2

我无法理解为什么这个程序永远等待,我正在使用我们正在调用 synchronized()、wait() 和 notify() 的 SAME 对象。按照我的理解,应该可以,不知道为什么不行。

关于为什么要永远等待的任何线索。

更新:

我对代码做了一些修改,UPDATE,它工作正常。

我还有点疑惑。 notify() 是否被线程调用,即使它没有锁定监视器,就像我更新代码后的情况一样。

事件顺序:

首先执行奇数线程,然后调用 wait() <-- 它释放监视器,现在处于睡眠模式。

甚至线程运行,打印消息,并调用 notify() <-- 在这里我没有清楚地理解。

Even线程调用notify()时,此时它有monitor,那么当它调用notify()时,是否仍然拥有monitor?

现在,在偶数线程调用notify()之后,奇数线程得到通知,因此它从休眠点开始执行。它正在执行一些操作并调用 notify(),此时我认为奇数线程不拥有监视器,它调用 notify()。所以,我的问题是,无论线程是否拥有监视器,notify() 的工作是否相同?

只有自己写代码的时候才真正明白这一点。我看了书,觉得我什么都懂了,看来我又回到原点了!

这里的问题很简单,就是两个线程都直接进入了等待状态。线程 1 获取 so,打印值然后等待。线程 2 然后获取 so,打印值然后等待。所以他们都在睡觉,因为没有人在那里通知他们。因此,一个简单的解决方法是在 so.wait() 之前执行 so.notify()。那么他们就不会无限期地等待。

编辑

奇数线程启动、执行然后等待。然后甚至线程启动、执行、通知并等待。 Even thread 持有监视器上的锁,直到它进入等待状态。

当偶数线程调用notify时,奇数线程唤醒并轮询锁。一旦偶数线程进入等待状态(&释放锁),那么奇数线程就可以获得锁。

如果偶数线程没有调用notify,那么奇数线程会继续休眠。偶数线程会去等待并释放锁。没有线程正在轮询或尝试获取锁,因此程序保持挂起状态。

documentation也提供了类似的解释。我希望这能消除你的疑虑。