文章轉載自:https://blog.csdn.net/weixin_41101173/article/details/79889464
sleep()和wait()方法都是Java中造成線程阻塞的方法。感興趣的讀者可以參見筆者之前的文章《Java中什麼方法導致線程阻塞》,裏面詳細講述了爲什麼Java要造成線程阻塞和Java中造成線程阻塞的幾種方法。
線程的生命週期
這是筆者在谷歌圖片中找到的一張簡單描述線程生命週期的圖片,可以看到,一個線程正常的生命週期中會經歷“創建”“就緒”“運行”“阻塞”“運行”“死亡”等幾個生命週期,其中“阻塞”是我們開發者非常需要關注的,因爲通過Java提供的阻塞方法,可以做到資源和線程最合理的調配,而其中sleep()和wait()方法就是Java中實現線程阻塞的關鍵性方法。
(2)sleep()和wait()方法的阻塞線程的場景
我們這裏,來看第二張圖片,這張圖片就比較詳細的向我們展現了一個線程的完整生命週期中發生的各種“事件”和調用的各種方法。我們來總結一下sleep()和wait()方法的阻塞場景。
①sleep()實現線程阻塞的方法,我們稱之爲“線程睡眠”,方式是超時等待,怎麼理解?就是sleep()通過傳入“睡眠時間”作爲方法的參數,時間一到就從“睡眠”中“醒來”;
②wait()方法實現線程阻塞的方法,我們稱之爲“線程等待”,方式有兩種:
1)和sleep()方法一樣,通過傳入“睡眠時間”作爲參數,時間到了就“醒了”;
2)不傳入時間,進行一次“無限期的等待”,只用通過notify()方法來“喚醒”。
二、sleep()和wait()的區別
通過上面兩幅圖的展示和筆者的相關描述,相信讀者你已經有幾分清楚了,sleep()和wait()方法的區別之一,就是實現線程阻塞的方式不一樣。
那麼sleep()和wait()二者還有一個很大的區別就是,二者“是否釋放同步鎖”不一樣。我們都知道,多線程開發中,爲了實現不同線程間的同步會採用同步鎖的方式——synchronized即在線程使用一個資源時爲其加鎖,這樣其他的線程便不能訪問那個資源了,直到解鎖後纔可以訪問。感興趣的讀者可以參考筆者之前文章《利用synchronized實現線程同步》裏面詳細介紹了多線程開發中利用synchronized實現線程同步的方式。
而使用sleep()和wait()兩種方法對於“CPU執行權”和“同步鎖”的方式不同:
①sleep()釋放CPU執行權,但不釋放同步鎖;
②wait()釋放CPU執行權,也釋放同步鎖,使得其他線程可以使用同步控制塊或者方法。
以上,就是sleep()和wait()方法的兩個關鍵性區別。
總結:綜上我們利用下表展示sleep()和wait()的所有區別:
ps:謝謝博友的更正。這裏進行改正: wait()爲Object基類中的方法,調用時亦需要捕獲異常,以下爲wait()可能拋出的異常:
PS:能用圖說明的問題,我們就用圖來說明,能用表說明的問題,我們就用表來說明,這樣理解起來非常清晰。