java之yield(),sleep(),wait()差別詳解-備忘筆記
部落格分類:java-->>線程
java線程sleep yieldrun start
1、sleep()
使目前線程(即調用該方法的線程)暫停執行一段時間,讓其他線程有機會繼續執行,但它并不釋放對象鎖。也就是說如果有synchronized同步快,其他線程仍然不能通路共享資料。注意該方法要捕捉異常。
例如有兩個線程同時執行(沒有synchronized)一個線程優先級為MAX_PRIORITY,另一個為MIN_PRIORITY,如果沒有Sleep()方法,隻有高優先級的線程執行完畢後,低優先級的線程才能夠執行;但是高優先級的線程sleep(500)後,低優先級就有機會執行了。
總之,sleep()可以使低優先級的線程得到執行的機會,當然也可以讓同優先級、高優先級的線程有執行的機會。
2、join()
join()方法使調用該方法的線程在此之前執行完畢,也就是等待該方法的線程執行完畢後再往下繼續執行。注意該方法也需要捕捉異常。
3、yield()
該方法與sleep()類似,隻是不能由使用者指定暫停多長時間,并且yield()方法隻能讓同優先級的線程有執行的機會。
4、wait()和notify()、notifyAll()
這三個方法用于協調多個線程對共享資料的存取,是以必須在synchronized語句塊内使用。synchronized關鍵字用于保護共享資料,阻止其他線程對共享資料的存取,但是這樣程式的流程就很不靈活了,如何才能在目前線程還沒退出synchronized資料塊時讓其他線程也有機會通路共享資料呢?此時就用這三個方法來靈活控制。
wait()方法使目前線程暫停執行并釋放對象鎖标示,讓其他線程可以進入synchronized資料塊,目前線程被放入對象等待池中。當調用notify()方法後,将從對象的等待池中移走一個任意的線程并放到鎖标志等待池中,隻有鎖标志等待池中線程能夠擷取鎖标志;如果鎖标志等待池中沒有線程,則notify()不起作用。
notifyAll()則從對象等待池中移走所有等待那個對象的線程并放到鎖标志等待池中。
注意 這三個方法都是java.lang.Object的方法。
二、run和start()
把需要處理的代碼放到run()方法中,start()方法啟動線程将自動調用run()方法,這個由java的記憶體機制規定的。并且run()方法必需是public通路權限,傳回值類型為void。
三、關鍵字synchronized
該關鍵字用于保護共享資料,當然前提條件是要厘清哪些資料是共享資料。每個對象都有一個鎖标志,當一個線程通路到該對象,被Synchronized修飾的資料将被"上鎖",阻止其他線程通路。目前線程通路完這部分資料後釋放鎖标志,其他線程就可以通路了。
四、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阻塞,那麼較低優先級線程隻能等待所有較高優先級的線程運作結束,方可有機會運作。