“在同步方法内调用等待是获取内在锁的简单方法”
这句话是假的,它是文档中的错误。
当线程进入同步方法时,它会获取固有锁。
同步方法中的线程被设置为锁的所有者并且处于RUNNABLE状态。
尝试进入锁定方法的任何线程都将变为BLOCKED。
当线程调用等待它释放当前对象锁(它保持所有锁与其他对象),而不是去WAITING状态。
当一些其他线程调用在同一对象上通知或notifyAll时,第一个线程将状态从WAITING更改为BLOCKED,
通知线程不会自动重新获取锁或变为RUNNABLE,实际上它必须争取与所有其他被阻止的线程的锁。
WAITING和BLOCKED状态都防止线程运行,但是它们是非常不同的。
WAITING线程必须通过一些其他线程的通知显式地转换为BLOCKED线程。
等待永远不会直接到RUNNABLE。
当RUNNABLE线程释放锁时(通过离开监视器或等待),BLOCKED线程中的一个自动占据它的位置。
所以总结一下,线程在进入同步方法时或在等待之后重新进入同步方法时获取锁。
public synchronized guardedJoy() {
// must get lock before entering here
while(!joy) {
try {
wait(); // releases lock here
// must regain the lock to reentering here
} catch (InterruptedException e) {}
}
System.out.println("Joy and efficiency have been achieved!");
}