Java信號量全解析
前言:
Semaphore(信號量) 是一個線程同步結(jié)構(gòu),用于在線程間傳遞信號,以避免出現(xiàn)信號丟失(譯者注:下文會具體介紹),或者像鎖一樣用于保護(hù)一個關(guān)鍵區(qū)域。自從5.0開始,jdk在java.util.concurrent包里提供了Semaphore 的官方實現(xiàn),因此大家不需要自己去實現(xiàn)Semaphore。但是還是很有必要去熟悉如何使用Semaphore及其背后的原理
內(nèi)容主題:
一、簡單的Semaphore實現(xiàn)
下面是一個信號量的簡單實現(xiàn):
public class Semaphore {
private boolean signal = false;
public synchronized void take() {
this.signal = true;
this.notify();1011}
public synchronized void release() throws InterruptedException{
while(!this.signal) wait();
this.signal = false;
}
}
Take方法發(fā)出一個被存放在Semaphore內(nèi)部的信號,而Release方法則等待一個信號,當(dāng)其接收到信號后,標(biāo)記位signal被清空,然后該方法終止。
使用這個semaphore可以避免錯失某些信號通知。用take方法來代替notify,release方法來代替wait。如果某線程在調(diào)用release等待之前調(diào)用take方法,那么調(diào)用release方法的線程仍然知道take方法已經(jīng)被某個線程調(diào)用過了,因為該Semaphore內(nèi)部保存了take方法發(fā)出的信號。而wait和notify方法就沒有這樣的功能。
當(dāng)用semaphore來產(chǎn)生信號時,take和release這兩個方法名看起來有點奇怪。這兩個名字來源于后面把semaphore當(dāng)做鎖的例子,后面會詳細(xì)介紹這個例子,在該例子中,take和release這兩個名字會變得很合理。
二、使用Semaphore來產(chǎn)生信號
下面的例子中,兩個線程通過Semaphore發(fā)出的信號來通知對方
Semaphore semaphore = new Semaphore();
SendingThread sender = new SendingThread(semaphore);
ReceivingThread receiver = new ReceivingThread(semaphore);
receiver.start();
sender.start();
public class SendingThread {
Semaphore semaphore = null;
public SendingThread(Semaphore semaphore){
this.semaphore = semaphore;
}
public void run(){
while(true){
//do something, then signal
this.semaphore.take();
}
}
}
public class RecevingThread {
Semaphore semaphore = null;
public ReceivingThread(Semaphore semaphore){
this.semaphore = semaphore;
}
public void run(){
while(true){
this.semaphore.release();
//receive signal, then do something...
}
}
}
三、可計數(shù)的Semaphore
上面提到的Semaphore的簡單實現(xiàn)并沒有計算通過調(diào)用take方法所產(chǎn)生信號的數(shù)量??梢园阉脑斐删哂杏嫈?shù)功能的Semaphore。下面是一個可計數(shù)的Semaphore的簡單實現(xiàn)。
public class CountingSemaphore {
private int signals = 0;
public synchronized void take() {
this.signals++;0809this.notify();
}
public synchronized void release() throws InterruptedException{
while(this.signals == 0) wait();
this.signals--;
}
}
四、有上限的Semaphore
上面的CountingSemaphore并沒有限制信號的數(shù)量。下面的代碼將CountingSemaphore改造成一個信號數(shù)量有上限的BoundedSemaphore。
public class BoundedSemaphore {
private int signals = 0;
private int bound = 0;
public BoundedSemaphore(int upperBound){
this.bound = upperBound;
}
public synchronized void take() throws InterruptedException{
while(this.signals == bound) wait();
this.signals++;
this.notify();
}
public synchronized void release() throws InterruptedException{
while(this.signals == 0) wait();
this.signals--;
this.notify();
}
}
在BoundedSemaphore中,當(dāng)已經(jīng)產(chǎn)生的信號數(shù)量達(dá)到了上限,take方法將阻塞新的信號產(chǎn)生請求,直到某個線程調(diào)用release方法后,被阻塞于take方法的線程才能傳遞自己的信號。
五、把Semaphore當(dāng)鎖來使用
當(dāng)信號量的數(shù)量上限是1時,Semaphore可以被當(dāng)做鎖來使用。通過take和release方法來保護(hù)關(guān)鍵區(qū)域。請看下面的例子:
BoundedSemaphore semaphore = new BoundedSemaphore(1);
...
semaphore.take();
try{
//critical section
} finally {
semaphore.release();
}
在前面的例子中,Semaphore被用來在多個線程之間傳遞信號,這種情況下,take和release分別被不同的線程調(diào)用。但是在鎖這個例子中,take和release方法將被同一線程調(diào)用,因為只允許一個線程來獲取信號(允許進(jìn)入關(guān)鍵區(qū)域的信號),其它調(diào)用take方法獲取信號的線程將被阻塞,知道第一個調(diào)用take方法的線程調(diào)用release方法來釋放信號。對release方法的調(diào)用永遠(yuǎn)不會被阻塞,這是因為任何一個線程都是先調(diào)用take方法,然后再調(diào)用release。
通過有上限的Semaphore可以限制進(jìn)入某代碼塊的線程數(shù)量。設(shè)想一下,在上面的例子中,如果BoundedSemaphore 上限設(shè)為5將會發(fā)生什么?意味著允許5個線程同時訪問關(guān)鍵區(qū)域,但是你必須保證,這個5個線程不會互相沖突。否則你的應(yīng)用程序?qū)⒉荒苷_\行。
必須注意,release方法應(yīng)當(dāng)在finally塊中被執(zhí)行。這樣可以保在關(guān)鍵區(qū)域的代碼拋出異常的情況下,信號也一定會被釋放。
以上就是Java信號量全解析的詳細(xì)內(nèi)容,更多關(guān)于Java信號量的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Intellij?IDEA根據(jù)maven依賴名查找它是哪個pom.xml引入的(圖文詳解)
這篇文章主要介紹了Intellij?IDEA根據(jù)maven依賴名查找它是哪個pom.xml引入的,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-08-08
Java并發(fā)容器之ConcurrentLinkedQueue詳解
這篇文章主要介紹了Java并發(fā)容器之ConcurrentLinkedQueue詳解,加鎖隊列的實現(xiàn)較為簡單,這里就略過,我們來重點來解讀一下非阻塞隊列,2023-12-12
從點到面, 下面我們來看下非阻塞隊列經(jīng)典實現(xiàn)類ConcurrentLinkedQueue,需要的朋友可以參考下
使用JavaConfig代替xml實現(xiàn)Spring配置操作
這篇文章主要介紹了使用JavaConfig代替xml實現(xiàn)Spring配置操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09
SpringBoot項目如何設(shè)置權(quán)限攔截器和過濾器
這篇文章主要介紹了使用lombok時如何自定義get、set方法問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-07-07
關(guān)于SpringGateway調(diào)用服務(wù) 接受不到參數(shù)問題
這篇文章主要介紹了關(guān)于SpringGateway調(diào)用服務(wù)接受不到參數(shù)問題,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-12-12

