Condition give the effect of having multiple wait-sets per object?

后端 未结 3 405
悲哀的现实
悲哀的现实 2021-01-07 10:28

I am reading about Condition in java.util.concurrent.locks.Condition .

Condition factors out the Object monitor methods (wait, notify and

3条回答
  •  太阳男子
    2021-01-07 10:48

    One Lock can be associated with many Conditions. Lock is an "object", each condition is a "waiting set". This allows for independent conditions sharing critical section. For example, consider bounded producers-consumers problem. One way to solve it is to have one lock that protects the queue, and two independent waiting sets: one for producers, waiting for slot to place item in the queue, other for consumers waiting for items to take. Using plain old synchronized and wait/notify API, best we can do is along these lines:

    • producer:

      synchronized (lock) {
          while (queue.isFull()) {
              lock.wait();
          }
          queue.put(sth);
          lock.notify();
      }
      
    • consumer:

      synchronized (lock) {
          while (queue.isEmpty() {
              lock.wait();
          }
          product = queue.take();
          lock.notify();
      }
      

    This has the disadvantage of waking up both producers and consumers on every change to the queue, even if it cannot possibly allow given thread to proceed (e.g. consumer is awoken when some other consumer takes item from the queue). Using Lock/Condition API we can implement solution that separates waiting consumers and producers, and hence reduce redundant wakeups and checking:

    Lock lock = new ReentrantLock();
    Condition hasPlace = lock.newCondition();
    Condition hasItems = lock.newCondition();
    
    • producer:

      lock.lock();
      try {
          while (queue.isFull()) {
              hasPlace.await();
          }
          queue.put(sth);
          hasItems.signal();
      } finally {
          lock.unlock();
      }
      
    • consumer:

      lock.lock();
      try {
          while (queue.isEmpty()) {
              hasItems.await();
          }
          product = queue.take();
          hasPlace.signal();
      } finally {
          lock.unlock();
      }
      

    This way, consumer waits for producers to produce some item (hasItems condition), and upon removing an item from the queue it notifies producers that there is an empty slot (hasPlace condition). Both conditions are associated with the same critical section (Lock), so we keep the usual exclusion and release-lock-while-waiting guarantees, while gaining the ability to separate waiting queues.

提交回复
热议问题