Unlike safety issues (which are about correctness), liveness hazards are about progress. Your program might be perfectly correct in theory, but under certain conditions, it may just stop moving forward.
Common types of liveness hazards include:
-
Deadlock β two or more threads wait forever for each other to release locks.
-
Livelock β threads keep reacting to each other but make no progress.
-
Starvation β a thread waits indefinitely because others are always chosen instead.
This post walks through 5 Java examples that illustrate these problems and shows how to avoid them by applying the recommended strategies.
A program will be free of lock-ordering deadlocks if all threads acquire the locks they need in a fixed global order.
1. LeftRightDeadlock.java β Simple Lock-Ordering Deadlock
This example demonstrates a basic deadlock caused by inconsistent locking order.
When two threads acquire the same locks in opposite order, they can end up waiting on each other forever.
π Full code:
LeftRightDeadlock.java
The deadlock can be analyzed with thread dump (kill -3 or press the Ctrl+\ key on Unix or Ctrl + Break on Windows).
2. DynamicOrderDeadlock.java β Randomized Lock-Ordering Deadlock
This example shows how deadlocks can happen dynamically in systems like banking transfers, where the locking order depends on randomly selected accounts.
If two threads select the same account pair in reverse order, they can deadlock.
π Full code:
DynamicOrderDeadlock.java
3. InduceLockOrder.java β Fixing Deadlock with Consistent Lock Ordering
To fix the problem in the previous example, we impose a global lock ordering by comparing the identity hash codes of the account objects.
This ensures that all threads acquire locks in the same order, eliminating circular wait and preventing deadlock.
π Full code:
InduceLockOrder.java
Invoking an alien method with lock held is asking for liveness trouble. The alien method might acquire other locks (risking deadlock) or block for an unexpectedly long time, stalling other threads that need the lock you hold.
4. CooperatingDeadlock.java β Mutual Calls Between Objects
In this example, we simulate a GPS system where a Taxi
updates its location and notifies a Dispatcher
when it becomes available. At the same time, the Dispatcher
retrieves locations of all taxis to update a GUI.
These circular calls between synchronized methods can easily result in deadlock if both threads hold one lock and wait on the other.
π Full code:
CooperatingDeadlock.java
Strive to use open calls throughout your program. Programs that rely on open calls are far easier to analyze for deadlock-freedom than those that allow calls to alien methods with locks held.
5. CooperatingNoDeadlock.java β Using Open Calls to Avoid Deadlock
This improved version breaks the circular locking by using open calls. The idea is simple: donβt call another synchronized method while holding a lock.
Similarly, the dispatcher copies the taxi set inside a synchronized block and then iterates over it outside the lock.
π Full code:
CooperatingNoDeadlock.java
Conclusion
Deadlocks are notoriously difficult to reproduce and debug. Fortunately, there are reliable strategies to avoid them:
-
Always acquire locks in a consistent order.
-
Use open callsβrelease your lock before calling into other objects.
-
Consider using
tryLock
or higher-level concurrent utilities. -
Avoid nested locking when possible.
No comments:
Post a Comment