為什么80%的碼農都做不了架構師?>>> ??
關于sleep和wait區別解析:
sleep只是釋放CPU資源,并不釋放資源鎖對象,wait是會釋放掉資源鎖對象。
比如,有個鎖對象object,線程1和線程2都會鎖住object對象。運行線程1,線程1中使用wait方法,這個時候,線程1就處于卡死狀態了,這時候我們運行線程2,線程2中使用sleep。這時候,如果資源鎖沒有釋放,線程2是不會被執行的,因為資源鎖被線程1占用,但是wait會釋放資源鎖,所以現象就是,線程1會執行,并且拿到資源鎖。
搞個例子來看下:
public static Object lock = new Object();
//開啟一個線程1 new Thread(() -> {//首先搞個鎖對象synchronized (lock) {System.out.println("線程1已拿到資源鎖對象");try {//注意此處的區別//lock.wait(5000);Thread.sleep(5000);} catch (Exception ex) {ex.printStackTrace();}System.out.println("線程1資源鎖準備釋放");}}).start();Thread.sleep(10);//開啟一個線程2 new Thread(() -> {//首先搞個鎖對象synchronized (lock) {System.out.println("線程2已拿到資源鎖對象");try {Thread.sleep(1000);} catch (Exception ex) {ex.printStackTrace();}System.out.println("線程2資源鎖準備釋放");} }).start();
注意線程1中注釋的代碼,我們先來看下都是sleep情況下的結果,如下圖,線程1資源釋放后,線程2才能拿到鎖繼續執行。
接下來我們把線程1中的sleep換成wati方法來看下結果,線程1拿到資源鎖后,調用wait會釋放資源鎖,這時線程2就可以拿到鎖的。如果線程1wait到時間后會繼續執行。
接下來測試下,wait喚醒后是否會拿到鎖后執行,還是不用拿鎖執行。調整下代碼,線程1中的wait去掉時間設定,改為手工喚醒。線程2中添加喚醒線程1的方法,然后運行程序看下結果。
//開啟一個線程1 new Thread(() -> {//首先搞個鎖對象synchronized (lock) {System.out.println("線程1已拿到資源鎖對象");try {//注意此處的區別lock.wait();//Thread.sleep(5000);} catch (Exception ex) {ex.printStackTrace();}System.out.println("線程1資源鎖準備釋放");}}).start();Thread.sleep(10);//開啟一個線程2 new Thread(() -> {//首先搞個鎖對象synchronized (lock) {System.out.println("線程2已拿到資源鎖對象");try {Thread.sleep(1000);//測試線程1是否會再次拿到鎖對象lock.notify();System.out.println("已喚醒線程1");Thread.sleep(3000);} catch (Exception ex) {ex.printStackTrace();}System.out.println("線程2資源鎖準備釋放");} }).start();
線程2喚醒線程1后,線程1并沒有馬上執行,而是等到線程2將資源鎖釋放后,線程1才開始執行。
?
讓線程睡眠今天發現還有另外一種方式,本質上其實還是Thread.sleep()
TimeUnit.SECONDS.sleep(4);
?
?