デッドロックとは、Java プログラムにおいて、リソースの競合により複数のスレッドが互いに待機し、その結果プロセスの実行を継続できなくなる状況を指します。デッドロックが発生すると、関係するスレッドの実行が継続できなくなり、プログラム全体が停止します。
相互排他条件 (相互排他): リソースが同じ時間は 1 つのスレッドによってのみ占有できます。スレッドがリソースを占有している場合、占有しているスレッドによってリソースが解放されるまで、他のスレッドはリソースにアクセスできません。
ホールド アンド ウェイト (ホールド アンド ウェイト): 少なくとも 1 つのリソースを保持している間、スレッドは他のスレッドが占有しているリソースを要求しようとします。これにより、スレッドは他のリソースを待機している間、すでに所有しているリソースを保持したままになります。
非プリエンプション条件 (プリエンプションなし): スレッドが占有しているリソースを他のスレッドがプリエンプトすることはできません。スレッドがリソースをアクティブに解放した場合にのみ、他のスレッドがリソースを取得できます。
循環待機 (循環待機): スレッド T1、T2、...、Tn のグループがあり、その中で T1 は、スレッドによって占有されるリソースを待機します。 T2、T2 は T3 を待機します。占有されているリソース、...、Tn は、T1 が占有しているリソースを待機し、循環待機関係を形成します。
スレッド間のリソース競合: 複数のスレッドが同時に共有リソースにアクセスすると、リソース競合が発生する可能性があります。 、デッドロックが発生します。
循環待機: スレッド間にはリソースに対する循環待機関係があり、各スレッドは他のスレッドがリソースを解放するのを待機します。
一貫性のない順序: スレッドがリソースを要求するときに、固定された順序で要求しないと、デッドロックが発生しやすくなります。
固定順序でリソースを要求する: すべてのスレッドが同じ順序でリソースを要求するようにします。デッドロックの可能性。
循環待機を回避する: スレッド間にリソースの循環待機関係がないことを確認します。
ロック タイムアウト設定を使用する: Java では、tryLock()
メソッドを使用してロック タイムアウトを設定し、タイムアウト後にロックが自動的に解放されるようにできます。デッドロックが発生する可能性を減らします。
次は Java デッドロックの例です:
public class DeadlockDemo { private static Object lock1 = new Object(); private static Object lock2 = new Object(); public static void main(String[] args) { new Thread(() -> { synchronized (lock1) { System.out.println("Thread 1: Holding lock 1"); try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("Thread 1: Waiting for lock 2"); synchronized (lock2) { System.out.println("Thread 1: Holding lock 1 & 2"); } } }).start(); new Thread(() -> { synchronized (lock2) { System.out.println("Thread 2: Holding lock 2"); try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("Thread 2: Waiting for lock 1"); synchronized (lock1) { System.out.println("Thread 2: Holding lock 1 & 2"); } } }).start(); } }
上の例では、スレッド 1 とスレッド 2 がそれぞれロックされています。 ロック1
とロック2
。しかし、相手がロックしているリソースを取得しようとすると、双方とも相手がリソースを解放するのを待っているため、デッドロックが発生します。
Java には、デッドロックの問題を検出および分析するためのツールとメソッドがいくつか用意されています。
jstack
ツールを使用します: jstack
は、スレッド スタック情報を分析するために使用できる Java 用のコマンド ライン ツールです。プログラムでデッドロックが発生した場合、jstack
を介してスレッドのステータスを表示し、どのスレッドがデッドロックしたかを判断できます。
ThreadMXBean
の使用: ThreadMXBean
は Java Management Extensions (JMX) の一部であり、デッドロックの検出に使用できます。簡単な例を次に示します:
import java.lang.management.ManagementFactory; import java.lang.management.ThreadInfo; import java.lang.management.ThreadMXBean; public class DeadlockDetector { public static void main(String[] args) { ThreadMXBean threadMXBean = ManagementFactory.getThreadMXBean(); long[] deadlockedThreads = threadMXBean.findDeadlockedThreads(); if (deadlockedThreads != null) { ThreadInfo[] threadInfos = threadMXBean.getThreadInfo(deadlockedThreads); for (ThreadInfo threadInfo : threadInfos) { System.out.println("Deadlocked thread: " + threadInfo.getThreadId() + " - " + threadInfo.getThreadName()); } } else { System.out.println("No deadlocked threads found."); } } }
以上がJavaのデッドロック問題に対処する方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。