Współbieżność Java - impas
Zakleszczenie opisuje sytuację, w której dwa lub więcej wątków jest zablokowanych na zawsze, czekając na siebie. Zakleszczenie występuje, gdy wiele wątków potrzebuje tych samych blokad, ale uzyskuje je w innej kolejności. Program wielowątkowy Java może cierpieć z powodu zakleszczenia, ponieważsynchronizedsłowo kluczowe powoduje, że wątek wykonawczy blokuje się podczas oczekiwania na blokadę lub monitor skojarzony z określonym obiektem. Oto przykład.
Przykład
public class TestThread {
public static Object Lock1 = new Object();
public static Object Lock2 = new Object();
public static void main(String args[]) {
ThreadDemo1 T1 = new ThreadDemo1();
ThreadDemo2 T2 = new ThreadDemo2();
T1.start();
T2.start();
}
private static class ThreadDemo1 extends Thread {
public void run() {
synchronized (Lock1) {
System.out.println("Thread 1: Holding lock 1...");
try {
Thread.sleep(10);
} catch (InterruptedException e) {}
System.out.println("Thread 1: Waiting for lock 2...");
synchronized (Lock2) {
System.out.println("Thread 1: Holding lock 1 & 2...");
}
}
}
}
private static class ThreadDemo2 extends Thread {
public void run() {
synchronized (Lock2) {
System.out.println("Thread 2: Holding lock 2...");
try {
Thread.sleep(10);
} catch (InterruptedException e) {}
System.out.println("Thread 2: Waiting for lock 1...");
synchronized (Lock1) {
System.out.println("Thread 2: Holding lock 1 & 2...");
}
}
}
}
}
Kiedy kompilujesz i wykonujesz powyższy program, znajdujesz sytuację zakleszczenia, a poniżej znajdują się dane wyjściowe programu -
Wynik
Thread 1: Holding lock 1...
Thread 2: Holding lock 2...
Thread 1: Waiting for lock 2...
Thread 2: Waiting for lock 1...
Powyższy program zawiesi się na zawsze, ponieważ żaden z wątków nie jest w stanie kontynuować i nie czeka na siebie, aby zwolnić blokadę, więc możesz wyjść z programu naciskając CTRL + C.
Przykład rozwiązania impasu
Zmieńmy kolejność blokady i uruchommy ten sam program, aby sprawdzić, czy oba wątki nadal na siebie czekają -
Przykład
public class TestThread {
public static Object Lock1 = new Object();
public static Object Lock2 = new Object();
public static void main(String args[]) {
ThreadDemo1 T1 = new ThreadDemo1();
ThreadDemo2 T2 = new ThreadDemo2();
T1.start();
T2.start();
}
private static class ThreadDemo1 extends Thread {
public void run() {
synchronized (Lock1) {
System.out.println("Thread 1: Holding lock 1...");
try {
Thread.sleep(10);
} catch (InterruptedException e) {}
System.out.println("Thread 1: Waiting for lock 2...");
synchronized (Lock2) {
System.out.println("Thread 1: Holding lock 1 & 2...");
}
}
}
}
private static class ThreadDemo2 extends Thread {
public void run() {
synchronized (Lock1) {
System.out.println("Thread 2: Holding lock 1...");
try {
Thread.sleep(10);
} catch (InterruptedException e) {}
System.out.println("Thread 2: Waiting for lock 2...");
synchronized (Lock2) {
System.out.println("Thread 2: Holding lock 1 & 2...");
}
}
}
}
}
Więc po prostu zmiana kolejności blokad zapobiega wejściu programu w sytuację zakleszczenia i kończy się następującym wynikiem -
Wynik
Thread 1: Holding lock 1...
Thread 1: Waiting for lock 2...
Thread 1: Holding lock 1 & 2...
Thread 2: Holding lock 1...
Thread 2: Waiting for lock 2...
Thread 2: Holding lock 1 & 2...
Powyższy przykład ma na celu jedynie wyjaśnienie koncepcji, jednak jest to koncepcja złożona i należy się do niej zagłębić, zanim opracujesz aplikacje, które radzą sobie z sytuacjami zakleszczenia.