Imagine a tester and a developer are working on two applications, an Android application and an iPhone application. The Android application is in its testing phase; it should be tested and the reported bugs should be fixed by the developer. The iPhone application needs to be developed from scratch; it should be coded by the developer and then tested for bugs. Imagine the tester starts testing the Android application and the developer starts working with the iPhone application. The Android application completes the testing phase and requests for the developer to fix the bugs. At the same time, the iPhone application completes its development phase and requests the tester to test it. The managers working with both the Android and iPhone applications refuse to release their resources (developer or tester) before their project completes. So both projects are waiting for the other project to complete—that is, waiting for the same set of resources. This causes a deadlock—these threads might wait forever. Here’s how this looks in code:
Code Snippet :
packagecom.gs.corejava.collections;classDeveloper {publicsynchronizedvoidfixBugs() {System.out.println("Fixing bugs.."); }publicsynchronizedvoidcode() {System.out.println("Coding.."); }}classTester {publicsynchronizedvoidtestAppIn() {System.out.println("Testing.."); }}/** * Already developed , Hence first testing then fix bugs by dev team * * @author momalhotra * */classAndroidAppimplementsRunnable {Developer developer;Tester tester; /** * @param developer * @param tester */publicAndroidApp(Developer developer,Tester tester) { super();this.developer= developer;this.tester= tester; }publicvoidrun() {synchronized (tester) {tester.testAppIn();developer.fixBugs(); } }}/** * Starting development , then will be given to testing team * * @author momalhotra * */classIphoneAppimplementsRunnable {Developer developer;Tester tester; /** * @param developer * @param tester */publicIphoneApp(Developer developer,Tester tester) { super();this.developer= developer;this.tester= tester; }publicvoidrun() {synchronized (developer) {developer.code();tester.testAppIn(); } }}publicclassDeadlock {publicstaticvoidmain(String[] args) {Developer mohit =newDeveloper();Tester neeti =newTester();Thread iphoneApp =newThread(new IphoneApp(mohit, neeti));Thread androidApp =newThread(new AndroidApp(mohit, neeti));iphoneApp.start();androidApp.start(); }}
Database Deadlocks
A more complicated situation in which deadlocks can occur, is a database transaction. A database transaction may consist of many SQL update requests. When a record is updated during a transaction, that record is locked for updates from other transactions, until the first transaction completes. Each update request within the same transaction may therefore lock some records in the database.
If multiple transactions are running at the same time that need to update the same records, there is a risk of them ending up in a deadlock.
For example
Transaction 1, request 1, locks record 1 for update
Transaction 2, request 1, locks record 2 for update
Transaction 1, request 2, tries to lock record 2 for update.
Transaction 2, request 2, tries to lock record 1 for update.
Since the locks are taken in different requests, and not all locks needed for a given transaction are known ahead of time, it is hard to detect or prevent deadlocks in database transactions.
Programmatically, You can detect the threads which have entered into deadlock condition and also you can retrieve the details about them. This can be done using ThreadMXBean interface of java.lang.Management package. You can go through the oracle docs of ThreadMXBean interface.
First, you have to get an instance of ThreadMXBean using getThreadMXBean() method of ManagementFactory, like this.
After getting an instance of ThreadMXBean, call findMonitorDeadlockedThreads() method on it. It returns an array of type long containing ids of all currently deadlocked threads.
1
longids[] = bean.findMonitorDeadlockedThreads();
After getting the ids of deadlocked threads, pass these ids to getThreadInfo() method of ThreadMXBean. It will return an array of ThreadInfo objects, where one ThreadInfo object contains the details of one deadlocked thread.