1.sleep()
創(chuàng)新互聯建站是一家專業(yè)提供崇明企業(yè)網站建設,專注與做網站、成都網站制作、HTML5、小程序制作等業(yè)務。10年已為崇明眾多企業(yè)、政府機構等服務。創(chuàng)新互聯專業(yè)網站建設公司優(yōu)惠進行中。
使當前線程(即調用該方法的線程)暫停執(zhí)行一段時間,讓其他線程有機會繼續(xù)執(zhí)行,但它并不釋放對象鎖。也就是如果有Synchronized同步塊,其他線程仍然不同訪問共享數據。注意該方法要捕獲異常
比如有兩個線程同時執(zhí)行(沒有Synchronized),一個線程優(yōu)先級為MAX_PRIORITY,另一個為MIN_PRIORITY,如果沒有Sleep()方法,只有高優(yōu)先級的線程執(zhí)行完成后,低優(yōu)先級的線程才能執(zhí)行;但當高優(yōu)先級的線程sleep(5000)后,低優(yōu)先級就有機會執(zhí)行了。
總之,sleep()可以使低優(yōu)先級的線程得到執(zhí)行的機會,當然也可以讓同優(yōu)先級、高優(yōu)先級的線程有執(zhí)行的機會。
2.join()
join()方法使調用該方法的線程在此之前執(zhí)行完畢,也就是等待調用該方法的線程執(zhí)行完畢后再往下繼續(xù)執(zhí)行。注意該方法也要捕獲異常。
3.yield()
它與sleep()類似,只是不能由用戶指定暫停多長時間,并且yield()方法只能讓同優(yōu)先級的線程有執(zhí)行的機會。
4.wait()和notify()、notifyAll()
這三個方法用于協調多個線程對共享數據的存取,所以必須在Synchronized語句塊內使用這三個方法。前面說過Synchronized這個關鍵字用于保護共享數據,阻止其他線程對共享數據的存取。但是這樣程序的流程就很不靈活了,如何才能在當前線程還沒退出Synchronized數據塊時讓其他線程也有機會訪問共享數據呢?此時就用這三個方法來靈活控制。
wait()方法使當前線程暫停執(zhí)行并釋放對象鎖標志,讓其他線程可以進入Synchronized數據塊,當前線程被放入對象等待池中。當調用 notify()方法后,將從對象的等待池中移走一個任意的線程并放到鎖標志等待池中,只有鎖標志等待池中的線程能夠獲取鎖標志;如果鎖標志等待池中沒有線程,則notify()不起作用。
notifyAll()則從對象等待池中移走所有等待那個對象的線程并放到鎖標志等待池中。
注意 這三個方法都是java.lang.Ojbect的方法!
2.run()和start()
這兩個方法應該都比較熟悉,把需要并行處理的代碼放在run()方法中,start()方法啟動線程將自動調用 run()方法,這是由Java的內存機制規(guī)定的。并且run()方法必須是public訪問權限,返回值類型為void。
3.關鍵字Synchronized
這個關鍵字用于保護共享數據,當然前提是要分清哪些數據是共享數據。每個對象都有一個鎖標志,當一個線程訪問該對象時,被Synchronized修飾的數據將被“上鎖”,阻止其他線程訪問。當前線程訪問完這部分數據后釋放鎖標志,其他線程就可以訪問了。
public ThreadTest implements Runnable{ public synchronized void run(){ for(int i=0;i<10;i++){ System.out.println(" " + i); } } public static void main(String[] args) { Runnable r1 = new ThreadTest(); Runnable r2 = new ThreadTest(); Thread t1 = new Thread(r1); Thread t2 = new Thread(r2); t1.start(); t2.start(); } } //以上這段程序中的 i 變量并不是共享數據,也就是這里的Synchronized關鍵字并未起作用。因為t1,t2兩個線程是兩個對象(r1,r2)的線程。//不同的對象其數據是不同的,所以r1和r2兩個對象的i變量是并不是共享數據。 當把代碼改成如下:Synchronized關鍵字才會起作用 Runnable r = new ThreadTest(); Thread t1 = new Thread(r); Thread t2 = new Thread(r); t1.start(); t2.start();
wait()和notify(),notifyAll()是Object類的方法,sleep()和yield()是Thread類的方法。
(1).常用的wait方法有wait()和wait(long timeout):
void wait() 在其他線程調用此對象的 notify() 方法或 notifyAll() 方法前,導致當前線程等待。
void wait(long timeout) 在其他線程調用此對象的 notify() 方法或 notifyAll() 方法,或者超過指定的時間量前,導致當前線程等待。
wait()后,線程會釋放掉它所占有的“鎖標志”,從而使線程所在對象中的其它synchronized數據可被別的線程使用。
wait()和notify()因為會對對象的“鎖標志”進行操作,所以它們必須在synchronized函數或synchronized block中進行調用。如果在non-synchronized函數或non-synchronized block中進行調用,雖然能編譯通過,但在運 行時會發(fā)生IllegalMonitorStateException的異常。
(2).Thread.sleep(long millis),必須帶有一個時間參數。
sleep(long)使當前線程進入停滯狀態(tài),所以執(zhí)行sleep()的線程在指定的時間內肯定不會被執(zhí)行;
sleep(long)可使優(yōu)先級低的線程得到執(zhí)行的機會,當然也可以讓同優(yōu)先級和高優(yōu)先級的線程有執(zhí)行的機會;
sleep(long)是不會釋放鎖標志的。
(3).yield()沒有參數。
sleep 方法使當前運行中的線程睡眼一段時間,進入不可運行狀態(tài),這段時間的長短是由程序設定的,yield 方法使當前線程讓出CPU占有權,但讓出的時間是不可設定的。
yield()也不會釋放鎖標志。
實際上,yield()方法對應了如下操作: 先檢測當前是否有相同優(yōu)先級的線程處于同可運行狀態(tài),如有,則把 CPU 的占有權交給此線程,否則繼續(xù)運行原來的線程。所以yield()方法稱為“退讓”,它把運行機會讓給了同等優(yōu)先級的其他線程。
sleep方法允許較低優(yōu)先級的線程獲得運行機會,但yield()方法執(zhí)行時,當前線程仍處在可運行狀態(tài),所以不可能讓出較低優(yōu)先級的線程些時獲得CPU占有權。 在一個運行系統中,如果較高優(yōu)先級的線程沒有調用 sleep 方法,又沒有受到 I/O阻塞,那么較低優(yōu)先級線程只能等待所有較高優(yōu)先級的線程運行結束,才有機會運行。
yield()只是使當前線程重新回到可執(zhí)行狀態(tài),所以執(zhí)行yield()的線程有可能在進入到可執(zhí)行狀態(tài)后馬上又被執(zhí)行。所以yield()只能使同優(yōu)先級的線程有執(zhí)行的機會。
以上這篇關于多線程常用方法以及對鎖的控制(詳解)就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持創(chuàng)新互聯。
文章標題:關于多線程常用方法以及對鎖的控制(詳解)
URL分享:http://www.rwnh.cn/article6/pgedog.html
成都網站建設公司_創(chuàng)新互聯,為您提供網站設計公司、虛擬主機、域名注冊、網站排名、微信小程序、網站策劃
聲明:本網站發(fā)布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創(chuàng)新互聯