data:image/s3,"s3://crabby-images/754a0/754a01d51973f1465c9fd004965001ec17e7895a" alt="DSA Takeover Cheatsheet"
Open-Source Internship opportunity by OpenGenus for programmers. Apply now.
Reading time: 25 minutes | Coding time: 10 minutes
📝 Table of Contents
What is Deadlock Synchronized keyword
Synchronized block
Deadlock Example
Deadlock Resolved
Questions
⭐ What is a Deadlock ?
The term Deadlock refers to a situation in multithreading, where two or more threads are blocked forever, waiting for each other.
It is a simply a condition where a set of threads are blocked because each thread is holding a resource and waiting for another resource acquired by some other thread.
Since, all the threads are waiting for each other to release the resource, the condition is called Deadlock.
data:image/s3,"s3://crabby-images/425cf/425cf3f2d53617cdf36453e36d79be464ce65278" alt="topic of image"
In the above image, Thread 1 is waiting for Thread 2 to relase some resource, while Thread 2 is waiting for Thread 1 to release some resource. Hence, a deadlock is created between them.
⭐ synchronized keyword -
In Java, we use synchronized keyword to make the class or method thread-safe.The keyword ensures that only one thread can access the resource at a given point of time.
Generally, In Java, Deadlock is caused because of the synchronized keyword, because it blocks the executing thread to wait for the lock.
⭐ synchronized block -
In Java, synchronized blocks are used for creating threads and synchronizing their tasks. Such blocks are marked with synchronized keyword.
A synchronized block in Java is synchronized on some object. The block ensures that only one thread executes inside it at a time. All other threads attempting to enter it are blocked until the thread inside it exits.
General form of a synchronized block -
synchronized(sync_object)
{
// Access shared variables and other
// shared resources
}
⭐ Example of a Deadlock in Java -
public class CreateDeadlock {
public static void main(String[] args) {
final String resource1 = "Resource 1";
final String resource2 = "Resource 2";
// Thread 1 tries to lock Resource 1 and then Resource 2
Thread t1 = new Thread() {
public void run() {
synchronized (resource1) {
System.out.println("Thread 1: I have locked Resource 1");
try { Thread.sleep(50);} catch (Exception e) {}
synchronized (resource2) {
System.out.println("Thread 1: I have locked Resource 2"); }
}
}
};
// Thread 2 tries to lock Resource 2 then Resource 1
Thread t2 = new Thread() {
public void run() {
synchronized (resource2) {
System.out.println("Thread 2: I have locked Resource 2 ");
try { Thread.sleep(50);} catch (Exception e) {}
synchronized (resource1) {
System.out.println("Thread 2: I have locked Resource 2");
}
}
}
};
t1.start();
t2.start();
}
}
Output :
Let's see step by step what is happening here -
- Thread 1 starts and acquires lock on resource 1.
- Thread 2 starts and acquires lock on resource 2.
- Thread 1 prints "Thread 1: I have locked Resource 1" and Thread 2 prints "Thread 2: I have locked Resource 2 ".
- Both of the threads wait for 50ms, so that both threads can be started if any of them is not.
- Thread 1 tries to take object lock of resource 2 but it is already acquired by Thread 2 so it waits till it become free. It will not release lock of Resource 1 until it gets lock of Resource 2.
- And same happens with thread 2. It tries to take object lock of resource 1 but it is already acquired by Thread 1 so it waits till it become free. It will not release lock of Resource 2 until it gets lock of Resource 1.
- Now, both threads are in wait state, and are waiting for each other to release the locks.
- As none of the thread is ready to release lock, so this is the Dead Lock condition.
- And hence when we run the program, it seems as if the program execution has stopped.
Diagrammatic representation of the Deadlock -
data:image/s3,"s3://crabby-images/ce9ba/ce9bac4204b244039a988f59f886afc42dbbba8f" alt="topic of image"
⭐ Resolving the deadlock -
To resolve this deadlock, we can simply change the order of acquiring the locks.
Let's try this -
public class ResolveDeadlock {
public static void main(String[] args) {
final String resource1 = "Resource 1";
final String resource2 = "Resource 2";
// Thread 1 tries to lock Resource 2 and then Resource 1
Thread t1 = new Thread() {
public void run() {
synchronized (resource2) {
System.out.println("Thread 1: I have locked Resource 2");
try { Thread.sleep(50);} catch (Exception e) {}
System.out.println("Thread 1: I am waiting for Resource 1");
synchronized (resource1) {
System.out.println("Thread 1: I have locked Resource 1 and 2");
}
}
}
};
// Thread 2 tries to lock Resource 2 and then Resource 1
Thread t2 = new Thread() {
public void run() {
synchronized (resource2) {
System.out.println("Thread 2: I have locked Resource 2 ");
try { Thread.sleep(50);} catch (Exception e) {}
System.out.println("Thread 2: I am waiting for Resource 1");
synchronized (resource1) {
System.out.println("Thread 2: I have locked Resource 1");
}
}
}
};
t1.start();
t2.start();
}
}
Output :
And our deadlock has been successfully resolved ! 🎉