wait 等待方法是让线程进入等待队列,使用方法是 obj.wait(); 这样当前线程就会暂停运行,并且进入obj的等待队列中,称作“线程正在obj上等待”。
如果线程想执行 wait 方法,线程必须拥有锁。如果线程进入等待队列,就会释放其实例的锁。
有两种形式的wait方法
一种是:接受毫秒数作为参数,含义与sleep中方法里接受参数的意思相同,都是指“在此期间暂停”,但是与sleep方法不同的是,wait 在暂停等待期间,对象锁是释放的,而sleep是拥抱着这把锁;还有一点不同的是,wait方法可以通过notify,notifyAll或者等待的时间到期,都可以令此线程接着执行。(时间参数值必须大于0,如果小于0,则会抛出 IllegalArgumentException运行异常,
如果时间参数为0 ,则会一直等待下去)
在接收参数的wait方法中,有两种分别是:
public final void wait(long timeout) throws InterruptedException
public final void wait(long timeout,int nanos)throws InterruptedException
1 public final void wait(long timeout, int nanos) throws InterruptedException { 2 if (timeout < 0) { 3 throw new IllegalArgumentException("timeout value is negative"); 4 } 5 6 if (nanos < 0 || nanos > 999999) { 7 throw new IllegalArgumentException( 8 "nanosecond timeout value out of range"); 9 } 10 11 if (nanos >= 500000 || (nanos != 0 && timeout == 0)) { 12 timeout++; 13 } 14 15 wait(timeout); 16 }
另一种是:wait 方法不接受任何参数,这种wait(),将无限等待下去,直到线程接收到nofity或者notifyAll的通知信息,才能继续执行。
摘自JDK 的API:
在没有被通知、中断或超时的情况下,线程还可以唤醒一个所谓的虚假唤醒 (spurious wakeup)。虽然这种情况在实践中很少发生,但是应用程序必须通过以下方式防止其发生,即对应该导致该线程被提醒的条件进行测试,如果不满足该条件,则继续等待。换句话说,等待应总是发生在循环中,如下面的示例:
synchronized (obj) {
while (<condition does not hold>)
obj.wait(timeout);
... // Perform action appropriate to condition
}
如果在当前线程等待通知之前或者正在等待通知时,任何线程中断了当前线程。在抛出此异常时,当前线程的中断状态 被清除,就会抛出中断异常InterruptedException