Last updated
Last updated
For synchronized keyword usage, when the thread could not get all resources, it will enter blocked state and could not do anything else.
Idea: Among the four conditions to break deadlock, three of them are possible. For the no preemption condition, it could be broken by proactively releasing its resources if not getting all necessary resources. There are three ways to achieve this:
Support interruption
Support timeout
Support trying to acquire lock without entering blocked state in failure case
Lock provides three method for implementing this
Requirements:
Allow multiple threads to read shared variables together.
Allow a single thread to write shared variable.
When a write operation is going on, no read operations will be required.
It does not support lock upgrade: if you get a read lock first, you could not get a write lock without releasing the read lock.
It supports lock downgrade: if you get a write lock first, you could get a read lock implicitly.
Support three types of lock
Write / pessimstic read
Share same semantics with ReadWriteLock
Difference: Needs to pass an additional parameter "stamp"
Optimistic read
For ReadWriteLock, when multiple threads are reading, no write operation is allowed at all; For StampedLock, when multiple threads are reading, a single thread is allowed to write.
StampedLock is a subclass of ReadWriteLock, and ReentrantReadWriteLock is also a subclass of ReadWriteLock. StampedLock is a non-reentrant lock.
It is suitable for the situation of more reading and less writing. If it is not the case, please use it with caution, the performance may not be as good as synchronized.
The pessimistic read lock and write lock of StampedLock do not support condition variables.
Never interrupt a blocked pessimistic read lock or write lock. If you call interrupt() of a blocked thread, it will cause the cpu to soar. If you want StampedLock to support interrupt operations, please use readLockInterruptibly( Pessimistic read lock) and writeLockInterruptibly (write lock).
Condition's await(), signal() and signalAll() are the same as wait()、notify()、notifyAll() from functional perspective.
References
Please see a sample of using Lock + Condition to implement producing-consuming pattern:
Criteria
synchronized (intrinsic lock)
ReentrantLock
Usage
implicitly acquire/release
explicitly acquire/release, best practice to put release inside finally
Competition strategy
Pessimistic. Will enter blocked state if failing to acquire resource
Optimistic. Will not enter blocked state by interruption, timeout and tryLock
Number of conditional variable
Single condition varialbe
Multiple condition variables
Fairness
Java's synchronized code block makes no guarantee about the sequence in which threads waiting to enter the synchronized block are allowed to enter.
Support both faiir and unfair lock. By default unfair.