案例:有一天小明和小麗兩個人去玩密室逃脫,在游戲過程中分別被關到不同的房間里,小明身上有能打開小麗房間的鑰匙,小麗身上有能打開小明房間的鑰匙。然而小明想要出去救小麗,就得有小麗身上的鑰匙,顯然他得不到;小麗想要出去救小明,就得有小明身上的鑰匙,顯然她也做不到。這種情況在我們程序界被稱為——死鎖。那具體什么是死鎖,為何出現(xiàn),當Java多線程遇到死鎖怎么辦,該怎么解決呢?
一、什么是死鎖?
在多線程環(huán)境中,多個進程可以競爭有限數量的資源。當進程申請資源時,如果此時沒有可用資源,則進程進入等待狀態(tài)。
有時,如果請求的資源被另一個等待進程占用,等待進程可能不再能夠改變狀態(tài)。這種情況稱為死鎖。
在 Java 中使用多線程可能會導致死鎖問題。死鎖會使程序卡住,不會繼續(xù)執(zhí)行。我們只能通過中止和重新啟動來重新執(zhí)行程序。
二、死鎖的原因
• 當前線程擁有其他線程需要的資源
• 當前線程正在等待另一個線程已經擁有的資源
• 不要放棄你擁有的資源
三、死鎖的必要條件
1.互斥
一個進程需要對分配的資源(如打印機)進行獨占控制,即一個資源在一段時間內只被一個進程占用。此時,如果另一個進程請求該資源,則請求進程只能等待。
2.不可剝奪
進程獲得的資源在用完之前不能被其他進程強行拿走,即只能由獲得資源的進程自己釋放(只能主動釋放)。
3.請求并保留
一個進程已經擁有了至少一個資源,但是又發(fā)起了一個新的資源請求,并且該資源已經被其他進程占用了。此時,請求進程被阻塞,但它獲得的資源不會被釋放。
4.循環(huán)等待
意思是進程死鎖發(fā)生后,進程和資源之間必然存在循環(huán)鏈。通俗的講,你在等我的資源,我在等你的資源,大家都在等。
四、 死鎖分類及解決方案
1.靜態(tài)順序死鎖
當線程形成相互等待資源的環(huán)時,就形成了順序死鎖lock-orderingdeadlock。當多個線程試圖以不同的順序獲取同一個鎖時,很容易形成順序死鎖。如果所有線程都以固定的順序來獲取鎖,就不會出現(xiàn)順序死鎖問題。
2. 動態(tài)鎖順序死鎖
由于方法的輸入參數是從外部傳入的,雖然這兩個參數在方法內部是按固定順序鎖定的,但是由于外部傳遞的順序不可控,即動態(tài)鎖定順序死鎖。
上面的例子告訴我們,交替獲取鎖會導致死鎖,而鎖是固定的。有時候鎖的執(zhí)行順序不是很清楚,參數導致執(zhí)行順序不同。
3.協(xié)作對象之間的死鎖
協(xié)作對象之間可能有多個鎖的獲取,但是這些多個鎖的獲取并不像 LeftRightDeadLock 或 transferMoney 中那么明顯,而且這兩個鎖不一定要在同一個方法中獲取。
如果在持有鎖的同時調用外部方法,那么需要警惕死鎖問題,因為在這個外部方法中可能會獲取其他鎖,或者阻塞時間過長,導致其他線程無法獲取當前保持鎖定時間。鎖。
當Java多線程遇到死鎖怎么辦?在上面的兩個例子中,兩個鎖是通過相同的方法獲取的。實際上,鎖不一定是通過相同的方法獲得的。更多關于“Java培訓”的問題,歡迎咨詢千鋒教育在線名師。千鋒已有十余年的培訓經驗,課程大綱更科學更專業(yè),有針對零基礎的就業(yè)班,有針對想提升技術的好程序員班,高品質課程助力你實現(xiàn)java程序員夢想。