wait() 与 notify() 和 notifyAll()
- wait():令当前线程挂起并放弃CPU、同步资源并等待,使别的线程可访问并修改共享资源,而当前线程排队等候其他线程调 用notify()或notifyAll()方法唤醒,唤醒后等待重新获得对监视器的所有权后才能继续执行。注意,notify()只是唤醒线程而已,不是说线程马上就能重新执行了,这时线程进入了同步阻塞状态,还要等待重新获得锁,最后获得cpu时间,才能重新执行。其实,我感觉wait的含义是本线程先释放这个锁,让其他线程先完成一些工作,然后其他拥有这个锁的线程完成任务后,再唤醒这个线程,就是等待拥有这个锁的线程的唤醒。其实,这个锁,大概率就是这个对象本身,因为要用synchronized嘛!!!
- notify():唤醒正在排队等待同步资源的线程中优先级最高者结束等待 。
- notifyAll ():唤醒正在排队等待资源的所有线程结束等待。
说明:
- 这三个方法只有在synchronized方法或synchronized代码块中才能使用,否则会报 java.lang.IllegalMonitorStateException异常。
- wait(),notify(),notifyAll()三个方法的调用者必须是同步代码块或同步方法中的同步监视器。否则,也会出现IllegalMonitorStateException异常。
- 因为这三个方法必须由锁对象调用,而任意对象都可以作为synchronized的同步锁, 因此这三个方法只能在Object类中声明。
详细说明:
wait()方法
- 在当前线程中调用方法: 对象名.wait() 。
- 使当前线程进入等待(某对象)状态 ,直到另一线程对该对象发出 notify (或notifyAll) 为止。
- 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)。
- 调用此方法后,当前线程将释放对象监控权 ,然后进入等待。
- 在当前线程被notify后,要重新获得监控权,然后从断点处继续代码的执行。
notify()/notifyAll()方法
- 在当前线程中调用方法: 对象名.notify()。
- 功能:唤醒等待该对象监控权的一个/所有线程。
- 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)。
面试题:sleep() 和 wait()的异同?
1.相同点:一旦执行方法,都可以使得当前的线程进入阻塞状态。
2.不同点:1)两个方法声明的位置不同:Thread类中声明sleep() , Object类中声明wait()
2)调用的要求不同:sleep()可以在任何需要的场景下调用。 wait()必须使用在同步代码块或同步方法中
3)关于是否释放同步监视器:如果两个方法都使用在同步代码块或同步方法中,sleep()不会释放锁,wait()会释放锁。