【问题标题】:Program gets halted: wait() and notify()程序停止:wait() 和 notify()
【发布时间】:2015-05-09 14:57:03
【问题描述】:

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

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

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

代码如下:

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

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

我无法理解为什么这个程序永远等待,我使用的是 SAME 对象,我们在该对象上调用了 synchronized()、wait() 和 notify()。根据我的理解,它应该可以工作,不知道为什么这不起作用。

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

更新:

我对代码进行了一些更改,更新,它工作正常。

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

事件顺序:

奇数线程首先被执行,然后它调用wait()

甚至线程运行,打印消息,并调用 notify()

当Even线程调用notify()时,此时它已经拥有了monitor,所以当它调用notify()时,是否仍然拥有monitor?

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

只有在编写代码时,才能真正理解这一点。我读了书,我觉得我什么都明白了,似乎我又回到了原点!

【问题讨论】:

  • 1) 在您确认有什么需要等待之前,请勿致电wait。 2)在某个地方,你必须存储你正在等待的东西,即接下来应该去哪个线程。没有它,线程就无法知道它是否应该等待或是否该工作了。

标签: java multithreading wait synchronized notify


【解决方案1】:

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

编辑

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

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

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

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

【讨论】:

  • +1。并且不要忘记在OddThread 中添加if(index&gt;=5) {so.notify();return;} 以唤醒EvenThread
  • 根据文档,wait() 解锁监视器,然后停止执行调用它的线程。它基于我已经编码。考虑一下,首先 OddThread 执行,然后它调用 wait(),此时它将解锁监视器并停止执行,然后 EvenThread 将执行,并且在遇到 wait() 时它将解锁监视器并停止执行。所以,我们必须在什么地方调用 notify()。有了这个模型,我们就没有任何机会通知()。我已经看到了 wait().. notify(); 的代码和其他线程也等待()..通知()..我不确定我在这里缺少什么!
  • 是的,wait() 表示解锁监视器并暂停执行。但是,这并不意味着其他线程开始执行。他们也处于“停止”模式。 Notify() 告诉他们现在重新开始执行。通常,程序是先写wait()语句,然后是notify(),但是,在执行时,你会发现一个线程进入wait(),而另一个线程跳过它并继续notify() . Wait() 语句通常在某些条件内。
  • 感谢@VineetRamachandran!我已经更新了问题,请看一下,我对 notify() 的行为 w.r.t 监视器/锁定有疑问。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-06-08
  • 1970-01-01
相关资源
最近更新 更多