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.