Thread.sleep()和Object.wait()的區別
首先,我們先來看看Thread.sleep()和Object.wait()的區別,這是一個爛大街的題目了,大家應該都能說上來兩點。
Thread.sleep()不會釋放占有的鎖,Object.wait()會釋放占有的鎖;Thread.sleep()必須傳入時間,Object.wait()可傳可不傳,不傳表示一直阻塞下去;Thread.sleep()到時間了會自動喚醒,然后繼續執行;Object.wait()不帶時間的,需要另一個線程使用Object.notify()喚醒; Object.wait()帶時間的,假如沒有被notify,到時間了會自動喚醒,這時又分好兩種情況,一是立即獲取到了鎖,線程自然會繼續執行;二是沒有立即獲取鎖,線程進入同步隊列等待獲取鎖;
其實,他們倆最大的區別就是Thread.sleep()不會釋放鎖資源,Object.wait()會釋放鎖資源。
Object.wait()和Condition.await()的區別
Object.wait()和Condition.await()的原理是基本一致的,不同的是Condition.await()底層是調用LockSupport.park()來實現阻塞當前線程的。
實際上,它在阻塞當前線程之前還干了兩件事,一是把當前線程添加到條件隊列中,二是“完全”釋放鎖,也就是讓state狀態變量變為0,然后才是調用LockSupport.park()阻塞當前線程。
Thread.sleep()和LockSupport.park()的區別
LockSupport.park()還有幾個兄弟方法——parkNanos()、parkUtil()等,我們這里說的park()方法統稱這一類方法。
從功能上來說,Thread.sleep()和LockSupport.park()方法類似,都是阻塞當前線程的執行,且都不會釋放當前線程占有的鎖資源; Thread.sleep()沒法從外部喚醒,只能自己醒過來; LockSupport.park()方法可以被另一個線程調用LockSupport.unpark()方法喚醒;Thread.sleep()方法聲明上拋出了InterruptedException中斷異常,所以調用者需要捕獲這個異常或者再拋出;LockSupport.park()方法不需要捕獲中斷異常;Thread.sleep()本身就是一個native方法;LockSupport.park()底層是調用的Unsafe的native方法;
Object.wait()和LockSupport.park()的區別
二者都會阻塞當前線程的運行,他們有什么區別呢? 經過上面的分析相信你一定很清楚了,真的嗎? 往下看!
Object.wait()方法需要在synchronized塊中執行; LockSupport.park()可以在任意地方執行;Object.wait()方法聲明拋出了中斷異常,調用者需要捕獲或者再拋出;LockSupport.park()不需要捕獲中斷異常;Object.wait()不帶超時的,需要另一個線程執行notify()來喚醒,但不一定繼續執行后續內容; LockSupport.park()不帶超時的,需要另一個線程執行unpark()來喚醒,一定會繼續執行后續內容;
park()/unpark()底層的原理是“二元信號量”,你可以把它相像成只有一個許可證的Semaphore,只不過這個信號量在重復執行unpark()的時候也不會再增加許可證,最多只有一個許可證。