java之yield(),sleep(),wait()區別詳解

一、sleep()java

使當前線程(即調用該方法的線程)暫停執行一段時間,讓其餘線程有機會繼續執行,但它並不釋放對象鎖。也就是說若是有synchronized同步快,其餘線程仍然不能訪問共享數據。注意該方法要捕捉異常。函數

例若有兩個線程同時執行(沒有synchronized)一個線程優先級爲MAX_PRIORITY,另外一個爲MIN_PRIORITY,若是沒有Sleep()方法,只有高優先級的線程執行完畢後,低優先級的線程纔可以執行;可是高優先級的線程sleep(500)後,低優先級就有機會執行了。線程

總之,sleep()可使低優先級的線程獲得執行的機會,固然也可讓同優先級、高優先級的線程有執行的機會。對象

 

二、join()內存

join()方法使調用該方法的線程在此以前執行完畢,也就是等待該方法的線程執行完畢後再往下繼續執行。注意該方法也須要捕捉異常。同步

 

三、yield()it

該方法與sleep()相似,只是不能由用戶指定暫停多長時間,而且yield()方法只能讓同優先級的線程有執行的機會。io

 

四、wait()和notify()、notifyAll()編譯

 

這三個方法用於協調多個線程對共享數據的存取,因此必須在synchronized語句塊內使用。synchronized關鍵字用於保護共享數據,阻止其餘線程對共享數據的存取,可是這樣程序的流程就很不靈活了,如何才能在當前線程還沒退出synchronized數據塊時讓其餘線程也有機會訪問共享數據呢?此時就用這三個方法來靈活控制。權限

wait()方法使當前線程暫停執行並釋放對象鎖標示,讓其餘線程能夠進入synchronized數據塊,當前線程被放入對象等待池中。當調用notify()方法後,將從對象的等待池中移走一個任意的線程並放到鎖標誌等待池中,只有鎖標誌等待池中線程可以獲取鎖標誌;若是鎖標誌等待池中沒有線程,則notify()不起做用。

notifyAll()則從對象等待池中移走全部等待那個對象的線程並放到鎖標誌等待池中。

注意 這三個方法都是java.lang.Object的方法。

 

 

2、run和start()

把須要處理的代碼放到run()方法中,start()方法啓動線程將自動調用run()方法,這個由java的內存機制規定的。而且run()方法必需是public訪問權限,返回值類型爲void。

 

3、關鍵字synchronized

該關鍵字用於保護共享數據,固然前提條件是要分清哪些數據是共享數據。每一個對象都有一個鎖標誌,當一個線程訪問到該對象,被Synchronized修飾的數據將被"上鎖",阻止其餘線程訪問。當前線程訪問完這部分數據後釋放鎖標誌,其餘線程就能夠訪問了。

 

 

4、wait()和notify(),notifyAll()是Object類的方法,sleep()和yield()是Thread類的方法。

(1)、經常使用的wait方法有wait()和wait(long timeout);

void wait() 在其餘線程調用此對象的 notify() 方法或者 notifyAll()方法前,致使當前線程等待。

void wait(long timeout)在其餘線程調用此對象的notify() 方法 或者 notifyAll()方法,或者超過指定的時間量前,致使當前線程等待。

wait()後,線程會釋放掉它所佔有的「鎖標誌」,從而使線程所在對象中的其餘shnchronized數據可被別的線程使用。

 

wait()h和notify()由於會對對象的「鎖標誌」進行操做,因此他們必需在Synchronized函數或者 synchronized block 中進行調用。若是在non-synchronized 函數或 non-synchronized block 中進行調用,雖然能編譯經過,但在運行時會發生IllegalMonitorStateException的異常。。

 

(2)、Thread.sleep(long millis)必須帶有一個時間參數。

sleep(long)使當前線程進入停滯狀態,因此執行sleep()的線程在指定的時間內確定不會被執行;

sleep(long)可以使優先級低的線程獲得執行的機會,固然也可讓同優先級的線程有執行的機會;

sleep(long)是不會釋放鎖標誌的。

 

(3)、yield()沒有參數

sleep 方法使當前運行中的線程睡眠一段時間,進入不能夠運行狀態,這段時間的長短是由程序設定的,yield方法使當前線程讓出CPU佔有權,但讓出的時間是不可設定的。

yield()也不會釋放鎖標誌。

實際上,yield()方法對應了以下操做;先檢測當前是否有相同優先級的線程處於同可運行狀態,若有,則把CPU的佔有權交給次線程,不然繼續運行原來的線程,因此yield()方法稱爲「退讓」,它把運行機會讓給了同等級的其餘線程。

 

sleep 方法容許較低優先級的線程得到運行機會,但yield()方法執行時,當前線程仍處在可運行狀態,因此不可能讓出較低優先級的線程此時獲取CPU佔有權。在一個運行系統中,若是較高優先級的線程沒有調用sleep方法,也沒有受到I/O阻塞,那麼較低優先級線程只能等待全部較高優先級的線程運行結束,方可有機會運行。

 

yield()只是使當前線程從新回到可執行狀態,全部執行yield()的線程有可能在進入到可執行狀態後立刻又被執行,因此yield()方法只能使同優先級的線程有執行的機會。

相關文章
相關標籤/搜索