java多线程中Object.wait()和Condition.await()是否会释放当前线程锁占有的锁

这个问题源自我问我们技术经理一个多线程问题时,他的回答让我迷惑。

我刚开始深入研究多线程,一直认为Object.wait()/Condition.await()让当前线程阻塞的同时,也会释放当前线程对该condition对象的锁。在之前的一些测试代码中也显示wait后,线程上的锁被释放了。但是我们经理却坚持当前线程会占用锁。

查看Object.wait()API 描述如下:

    Causes the current thread to wait until another thread invokes the notify() method or the notifyAll() method for this object. In other words, this method behaves exactly as if it simply performs the call wait(0). 

    The current thread must own this object's monitor. The thread releases ownership of this monitor and waits until another thread notifies threads waiting on this object's monitor to wake up either through a call to the notify method or the notifyAll method. The thread then waits until it can re-obtain ownership of the monitor and resumes execution. 

其中“the thread releases ownership of this monitor” 说道当前线程会释放这个对象监控器的所有权。

问题一:这里的monitor怎么理解?监视器(monitor)和锁是什么关系?
这个monitor就是一个术语,或者是一个特殊的类(只包含私有域),但是在java中并没有严格遵守这个概念。个人认为可以简单的理解为锁。

同样的,Condition.await()方法中也有类似描述。

    The lock associated with this Condition is atomically released and the current thread becomes disabled for thread scheduling purposes...

也说道,调用await()有,这个条件对象关联的锁被“原子级地”释放。。。

问题二:这个原子级的释放是什么意思?
“原子级”其实就是为了保证一个操作的完整性,原子级的释放保证了一个原子操作不会因为线程的突然挂起或者说阻塞而破坏这次操作。

这都能说明调用wait()或者await()后,当前线程会释放该条件对象关联的锁吧?!但是我们经理说不会释放又是什么意思?是我的理解太浅显么?还是我们经理错了?

希望多线程的资深前辈能不吝赐教!不胜感激

阅读 15.7k
2 个回答

你的经理错了 或者你们的沟通有问题吧. 

两个线程用object1wait/notify, 是这样:

thread1 得到object1 的 monitor, 调用 object1.wait() 
  - 释放object1 的 monitor, thread1 wait;

thread2 得到 object1 的 monitor, 调用 object1.notify() 
  - 激活thread1, 释放object1 的 monitor;

thread1 得到 object1 的 monitor, 从object1.wait()返回, thread1接着执行.

关于monitor, 这个是多进程/线程同步的 一个术语, 见:
Operating Systems Design and Implementation, Third Edition
section 2.2

A monitor is a collection of procedures, variables, and data
structures that are all grouped together in a special kind of module
or package. Processes may call the procedures in a monitor whenever
they want to, but they cannot directly access the monitor's internal
data structures from procedures declared outside the monitor. This
rule, which is common in modern object-oriented languages such as
Java, was relatively unusual for its time, although objects can be
traced back to Simula 67.

In all cases, before this method can return the current thread must

  • re-acquire the lock associated with this condition. When the

  • thread returns it is guaranteed to hold this lock.

    会释放,其他线程执行Condition.signal(),之前的线程会重新获得锁,继续执行,

    AbstractQueuedSynchronizer.java 第2040行,释放锁

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
宣传栏