If I just use synchronized, not wait/notify method, will it still keep thread-safe ?
What's the difference ?
Thx in advance.
If I just use synchronized, not wait/notify method, will it still keep thread-safe ?
What's the difference ?
Thx in advance.
so after just being embarrassed in an interview question on this I decided to look it up and understand it again for 1 billionth time lol.
synchronized block makes the code thread safe. No doubt about that. When wait() and notify() or notifyAll() come in is where your trying to write more efficient code. For example if you have a list of items that multiple threads share then if u put it in synchronized block of a monitor then threads threads will constantly jump in and run the code back and forth, back and fort during context switches......even with an empty list!
The wait() is hence used on the monitor (the object inside the synchronized(..)) as a mechanism to to tell all threads to chill out and stop using cpu cycles until further notice or notifyAll().
so something like:
Synchronised block is used, if 2 threads of "same object" tries to accquire the lock. Since object class holds the lock, it knows who to give. Whereas, if 2 threads(say t2 and t4) of 2 objects( t1 & t2 of obj1 and t3 & t4 of obj 2) try to acquire the lock, obj1 would be unaware of obj2's lock and obj2 would be unaware of obj1's lock. Hence wait and notify methods are used.
eg:
Two threads t1 and t2 belongs to same object, hence synchronization works fine here. Whereas,
When you run the above program, synchronisation does not work since each thread belong to different object, Hence you should use wait and notify here.
Using
synchronized
makes a method / block accessible by only on thread at a time. So, yes, it's thread-safe.The two concepts are combined, not mutually-exclusive. When you use
wait()
you need to own the monitor on that object. So you need to havesynchronized(..)
on it before that. Using.wait()
makes the current thread stop until another thread calls.notify()
on the object it waits on. This is an addition tosynchronized
, which just ensures that only one thread will enter a block/method.Making method as synchronized has two effects:
Second, when a synchronized method exits, it automatically establishes a happens-before relationship with any subsequent invocation of a synchronized method for the same object. This guarantees that changes to the state of the object are visible to all threads.
synchronization help you to guard the critical code.
If you want to establish communication between multiple threads, you have to use wait() and notify()/notifyAll()
wait()
: Causes the current thread to wait until another thread invokes the notify() method or the notifyAll() method for this object.notify()
: Wakes up a single thread that is waiting on this object's monitor. If any threads are waiting on this object, one of them is chosen to be awakened.notifyAll()
:Wakes up all threads that are waiting on this object's monitor. A thread waits on an object's monitor by calling one of the wait methods.Simple use case for using wait() and notify() : Producer and Consumer problem.
Consumer thread has to wait till Producer thread produce data. wait() and notify() are useful in above scenario. Over a period of time, better alternatives have been introduced. Refer to this high level concurrency tutorial page.
In simple terms:
Use
synchronized
to guard protect critical section of your data and guard your code.Use
wait()
andnotify()
along with synchronization if you want to establish communication between multiple threads in safe manner, which are interdependent on each other.Related SE questions:
What does 'synchronized' mean?
A simple scenario using wait() and notify() in java
Effective Java item 69: "Given the difficulty of using wait and notify correctly, you should use the higher-level concurrency utilities instead."
Avoid using wait() and notify(): use
synchronized
, or other utilities from java.util.concurrent, when possible.wait/notify is required when you want to wait for some condition (e.g. user input) INSIDE a synchronized block.
Typical usage:
Let's assume that you don't use wait(). Then, you have to implement busy loop polling the condition that you want, which is bad for performance.
Important note: Even though a thread is awaken by notify() or notifyAll() from other thread, the awaken thread does NOT guaranteed to immediately resume its execution. If there were other threads awaiting to execute a synchronized block on the same object, then the awaken thread should compete with the threads.