關(guān)于多線(xiàn)程常用方法以及對(duì)鎖的控制(詳解)
1.sleep()
使當(dāng)前線(xiàn)程(即調(diào)用該方法的線(xiàn)程)暫停執(zhí)行一段時(shí)間,讓其他線(xiàn)程有機(jī)會(huì)繼續(xù)執(zhí)行,但它并不釋放對(duì)象鎖。也就是如果有Synchronized同步塊,其他線(xiàn)程仍然不同訪(fǎng)問(wèn)共享數(shù)據(jù)。注意該方法要捕獲異常
比如有兩個(gè)線(xiàn)程同時(shí)執(zhí)行(沒(méi)有Synchronized),一個(gè)線(xiàn)程優(yōu)先級(jí)為MAX_PRIORITY,另一個(gè)為MIN_PRIORITY,如果沒(méi)有Sleep()方法,只有高優(yōu)先級(jí)的線(xiàn)程執(zhí)行完成后,低優(yōu)先級(jí)的線(xiàn)程才能執(zhí)行;但當(dāng)高優(yōu)先級(jí)的線(xiàn)程sleep(5000)后,低優(yōu)先級(jí)就有機(jī)會(huì)執(zhí)行了。
總之,sleep()可以使低優(yōu)先級(jí)的線(xiàn)程得到執(zhí)行的機(jī)會(huì),當(dāng)然也可以讓同優(yōu)先級(jí)、高優(yōu)先級(jí)的線(xiàn)程有執(zhí)行的機(jī)會(huì)。
2.join()
join()方法使調(diào)用該方法的線(xiàn)程在此之前執(zhí)行完畢,也就是等待調(diào)用該方法的線(xiàn)程執(zhí)行完畢后再往下繼續(xù)執(zhí)行。注意該方法也要捕獲異常。
3.yield()
它與sleep()類(lèi)似,只是不能由用戶(hù)指定暫停多長(zhǎng)時(shí)間,并且yield()方法只能讓同優(yōu)先級(jí)的線(xiàn)程有執(zhí)行的機(jī)會(huì)。
4.wait()和notify()、notifyAll()
這三個(gè)方法用于協(xié)調(diào)多個(gè)線(xiàn)程對(duì)共享數(shù)據(jù)的存取,所以必須在Synchronized語(yǔ)句塊內(nèi)使用這三個(gè)方法。前面說(shuō)過(guò)Synchronized這個(gè)關(guān)鍵字用于保護(hù)共享數(shù)據(jù),阻止其他線(xiàn)程對(duì)共享數(shù)據(jù)的存取。但是這樣程序的流程就很不靈活了,如何才能在當(dāng)前線(xiàn)程還沒(méi)退出Synchronized數(shù)據(jù)塊時(shí)讓其他線(xiàn)程也有機(jī)會(huì)訪(fǎng)問(wèn)共享數(shù)據(jù)呢?此時(shí)就用這三個(gè)方法來(lái)靈活控制。
wait()方法使當(dāng)前線(xiàn)程暫停執(zhí)行并釋放對(duì)象鎖標(biāo)志,讓其他線(xiàn)程可以進(jìn)入Synchronized數(shù)據(jù)塊,當(dāng)前線(xiàn)程被放入對(duì)象等待池中。當(dāng)調(diào)用 notify()方法后,將從對(duì)象的等待池中移走一個(gè)任意的線(xiàn)程并放到鎖標(biāo)志等待池中,只有鎖標(biāo)志等待池中的線(xiàn)程能夠獲取鎖標(biāo)志;如果鎖標(biāo)志等待池中沒(méi)有線(xiàn)程,則notify()不起作用。
notifyAll()則從對(duì)象等待池中移走所有等待那個(gè)對(duì)象的線(xiàn)程并放到鎖標(biāo)志等待池中。
注意 這三個(gè)方法都是java.lang.Ojbect的方法!
2.run()和start()
這兩個(gè)方法應(yīng)該都比較熟悉,把需要并行處理的代碼放在run()方法中,start()方法啟動(dòng)線(xiàn)程將自動(dòng)調(diào)用 run()方法,這是由Java的內(nèi)存機(jī)制規(guī)定的。并且run()方法必須是public訪(fǎng)問(wèn)權(quán)限,返回值類(lèi)型為void。
3.關(guān)鍵字Synchronized
這個(gè)關(guān)鍵字用于保護(hù)共享數(shù)據(jù),當(dāng)然前提是要分清哪些數(shù)據(jù)是共享數(shù)據(jù)。每個(gè)對(duì)象都有一個(gè)鎖標(biāo)志,當(dāng)一個(gè)線(xiàn)程訪(fǎng)問(wèn)該對(duì)象時(shí),被Synchronized修飾的數(shù)據(jù)將被“上鎖”,阻止其他線(xiàn)程訪(fǎng)問(wèn)。當(dāng)前線(xiàn)程訪(fǎng)問(wèn)完這部分?jǐn)?shù)據(jù)后釋放鎖標(biāo)志,其他線(xiàn)程就可以訪(fǎng)問(wèn)了。
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 變量并不是共享數(shù)據(jù),也就是這里的Synchronized關(guān)鍵字并未起作用。因?yàn)閠1,t2兩個(gè)線(xiàn)程是兩個(gè)對(duì)象(r1,r2)的線(xiàn)程。//不同的對(duì)象其數(shù)據(jù)是不同的,所以r1和r2兩個(gè)對(duì)象的i變量是并不是共享數(shù)據(jù)。
當(dāng)把代碼改成如下:Synchronized關(guān)鍵字才會(huì)起作用
Runnable r = new ThreadTest();
Thread t1 = new Thread(r);
Thread t2 = new Thread(r);
t1.start();
t2.start();
wait()和notify(),notifyAll()是Object類(lèi)的方法,sleep()和yield()是Thread類(lèi)的方法。
(1).常用的wait方法有wait()和wait(long timeout):
void wait() 在其他線(xiàn)程調(diào)用此對(duì)象的 notify() 方法或 notifyAll() 方法前,導(dǎo)致當(dāng)前線(xiàn)程等待。
void wait(long timeout) 在其他線(xiàn)程調(diào)用此對(duì)象的 notify() 方法或 notifyAll() 方法,或者超過(guò)指定的時(shí)間量前,導(dǎo)致當(dāng)前線(xiàn)程等待。
wait()后,線(xiàn)程會(huì)釋放掉它所占有的“鎖標(biāo)志”,從而使線(xiàn)程所在對(duì)象中的其它synchronized數(shù)據(jù)可被別的線(xiàn)程使用。
wait()和notify()因?yàn)闀?huì)對(duì)對(duì)象的“鎖標(biāo)志”進(jìn)行操作,所以它們必須在synchronized函數(shù)或synchronized block中進(jìn)行調(diào)用。如果在non-synchronized函數(shù)或non-synchronized block中進(jìn)行調(diào)用,雖然能編譯通過(guò),但在運(yùn) 行時(shí)會(huì)發(fā)生IllegalMonitorStateException的異常。
(2).Thread.sleep(long millis),必須帶有一個(gè)時(shí)間參數(shù)。
sleep(long)使當(dāng)前線(xiàn)程進(jìn)入停滯狀態(tài),所以執(zhí)行sleep()的線(xiàn)程在指定的時(shí)間內(nèi)肯定不會(huì)被執(zhí)行;
sleep(long)可使優(yōu)先級(jí)低的線(xiàn)程得到執(zhí)行的機(jī)會(huì),當(dāng)然也可以讓同優(yōu)先級(jí)和高優(yōu)先級(jí)的線(xiàn)程有執(zhí)行的機(jī)會(huì);
sleep(long)是不會(huì)釋放鎖標(biāo)志的。
(3).yield()沒(méi)有參數(shù)。
sleep 方法使當(dāng)前運(yùn)行中的線(xiàn)程睡眼一段時(shí)間,進(jìn)入不可運(yùn)行狀態(tài),這段時(shí)間的長(zhǎng)短是由程序設(shè)定的,yield 方法使當(dāng)前線(xiàn)程讓出CPU占有權(quán),但讓出的時(shí)間是不可設(shè)定的。
yield()也不會(huì)釋放鎖標(biāo)志。
實(shí)際上,yield()方法對(duì)應(yīng)了如下操作: 先檢測(cè)當(dāng)前是否有相同優(yōu)先級(jí)的線(xiàn)程處于同可運(yùn)行狀態(tài),如有,則把 CPU 的占有權(quán)交給此線(xiàn)程,否則繼續(xù)運(yùn)行原來(lái)的線(xiàn)程。所以yield()方法稱(chēng)為“退讓”,它把運(yùn)行機(jī)會(huì)讓給了同等優(yōu)先級(jí)的其他線(xiàn)程。
sleep方法允許較低優(yōu)先級(jí)的線(xiàn)程獲得運(yùn)行機(jī)會(huì),但yield()方法執(zhí)行時(shí),當(dāng)前線(xiàn)程仍處在可運(yùn)行狀態(tài),所以不可能讓出較低優(yōu)先級(jí)的線(xiàn)程些時(shí)獲得CPU占有權(quán)。 在一個(gè)運(yùn)行系統(tǒng)中,如果較高優(yōu)先級(jí)的線(xiàn)程沒(méi)有調(diào)用 sleep 方法,又沒(méi)有受到 I/O阻塞,那么較低優(yōu)先級(jí)線(xiàn)程只能等待所有較高優(yōu)先級(jí)的線(xiàn)程運(yùn)行結(jié)束,才有機(jī)會(huì)運(yùn)行。
yield()只是使當(dāng)前線(xiàn)程重新回到可執(zhí)行狀態(tài),所以執(zhí)行yield()的線(xiàn)程有可能在進(jìn)入到可執(zhí)行狀態(tài)后馬上又被執(zhí)行。所以yield()只能使同優(yōu)先級(jí)的線(xiàn)程有執(zhí)行的機(jī)會(huì)。
以上這篇關(guān)于多線(xiàn)程常用方法以及對(duì)鎖的控制(詳解)就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
新版本Spring中l(wèi)ombok失效的問(wèn)題解決
Lombok是一個(gè)Java庫(kù),通過(guò)添加注解簡(jiǎn)化代碼編寫(xiě),本文主要介紹了新版本Spring中l(wèi)ombok失效的問(wèn)題解決,感興趣的可以了解一下2025-01-01
maven依賴(lài)關(guān)系中的<scope>provided</scope>使用詳解
這篇文章主要介紹了maven依賴(lài)關(guān)系中的<scope>provided</scope>使用詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07
將InputStream轉(zhuǎn)化為base64的實(shí)例
這篇文章主要介紹了將InputStream轉(zhuǎn)化為base64的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12
springboot在filter中如何用threadlocal存放用戶(hù)身份信息
這篇文章主要介紹了springboot中在filter中如何用threadlocal存放用戶(hù)身份信息,本文章主要描述通過(guò)springboot的filter類(lèi),在過(guò)濾器中設(shè)置jwt信息進(jìn)行身份信息保存的方法,需要的朋友可以參考下2024-07-07
SpringBoot打包成Docker鏡像的幾種實(shí)現(xiàn)方式
Spring Boot是一個(gè)用于構(gòu)建獨(dú)立的、可執(zhí)行的Spring應(yīng)用程序的框架,結(jié)合使用Spring Boot和Docker,可以方便地將應(yīng)用程序部署到不同的環(huán)境中本文,主要介紹了SpringBoot打包成Docker鏡像的幾種實(shí)現(xiàn)方式,感興趣的可以了解一下2024-01-01

