理解此问题先修知识:
synchronized 的含义:
- Java中每一个对象都可以成为一个监视器(Monitor), 该Monitor由一个锁(lock), 一个等待队列(waiting
queue ), 一个入口队列( entry queue). - 对于一个对象的方法, 如果没有synchronized关键字, 该方法可以被任意数量的线程,在任意时刻调用。
- 对于添加了synchronized关键字的方法,任意时刻只能被唯一的一个获得了对象实例锁的线程调用。
- synchronized用于实现多线程的同步操作
wait()功用
- wait(), notify(), notifyAll() 和 synchonized 需要搭配使用, 用于线程同步
- wait()总是在一个循环中被调用,挂起当前线程来等待一个条件的成立。 Wait调用会一直等到其他线程调用notifyAll()时才返回。
- 当一个线程在执行synchronized 的方法内部,调用了wait()后, 该线程会释放该对象的锁,
然后该线程会被添加到该对象的等待队列中(waiting queue), 只要该线程在等待队列中, 就会一直处于闲置状态, 不会被调度执行。
要注意wait()方法会强迫线程先进行释放锁操作,所以在调用wait()时,
该线程必须已经获得锁,否则会抛出异常。由于wait()在synchonized的方法内部被执行, 锁一定已经获得, 就不会抛出异常了。
notify()的功用
- wait(), notify(), notifyAll() 和 synchonized 需要搭配使用, 用于线程同步
- 当一个线程调用一个对象的notify()方法时, 调度器会从所有处于该对象等待队列(waiting queue)的线程中取出任意一个线程,
将其添加到入口队列( entry queue) 中. 然后在入口队列中的多个线程就会竞争对象的锁, 得到锁的线程就可以继续执行。
如果等待队列中(waiting queue)没有线程, notify()方法不会产生任何作用 - notifyAll() 和notify()工作机制一样, 区别在于notifyAll()会将等待队列(waiting
queue)中所有的线程都添加到入口队列中(entry queue) - 注意, notifyAll()比notify()更加常用, 因为notify()方法只会唤起一个线程,
且无法指定唤醒哪一个线程,所以只有在多个执行相同任务的线程在并发运行时, 我们不关心哪一个线程被唤醒时,才会使用notify()
正确写法
// 线程 A 的代码
synchronized(obj_A)
{
while(!condition){
obj_A.wait();
}
// do something
}
// 线程 B 的代码
synchronized(obj_A)
{
if(!condition){
// do something ...
condition = true;
obj_A.notify();
}
}
————————————————
版权声明:本文为CSDN博主「萧萧冷」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/lengxiao1993/article/details/52296220