If one Googles for ‘difference between notify() and notifyAll()‘ then a lot of explanations will pop up (leaving apart the javadoc paragraphs). It all boils down to the number of waiting threads being waken up: one in notify() and all in notifyAll().
However (if I do understand the difference between these methods right), only one thread is always selected for further monitor acquisition; in the first case the one selected by the VM, in the second case the one selected by the system thread scheduler. The exact selection procedures for both of them (in the general case) are not known to the programmer.
What’s the useful difference between notify() and notifyAll() then? Am I missing something?
That is not correct.
o.notifyAll()wakes all of the threads that are blocked ino.wait()calls. The threads are only allowed to return fromo.wait()one-by-one, but they each will get their turn.Simply put, it depends on why your threads are waiting to be notified. Do you want to tell one of the waiting threads that something happened, or do you want to tell all of them at the same time?
In some cases, all waiting threads can take useful action once the wait finishes. An example would be a set of threads waiting for a certain task to finish; once the task has finished, all waiting threads can continue with their business. In such a case you would use notifyAll() to wake up all waiting threads at the same time.
Another case, for example mutually exclusive locking, only one of the waiting threads can do something useful after being notified (in this case acquire the lock). In such a case, you would rather use notify(). Properly implemented, you could use notifyAll() in this situation as well, but you would unnecessarily wake threads that can’t do anything anyway.
In many cases, the code to await a condition will be written as a loop:
That way, if an
o.notifyAll()call wakes more than one waiting thread, and the first one to return from theo.wait()makes leaves the condition in the false state, then the other threads that were awakened will go back to waiting.