Java實現(xiàn)限定時間CountDownLatch并行場景
業(yè)務(wù)場景:
一個用戶數(shù)據(jù)接口,要求在20ms內(nèi)返回數(shù)據(jù),它的調(diào)用邏輯復雜,關(guān)聯(lián)接口多,需要從3個接口匯總數(shù)據(jù),這些匯總接口最小耗時也需要16ms,全部匯總接口最優(yōu)狀態(tài)耗時需要16ms*3=48ms
解決方案:
使用并行調(diào)用接口,通過多線程同時獲取結(jié)果集,最后進行結(jié)果整合。在這種場景下,使用concurrent包的CountDownLatch完成相關(guān)操作。CountDownLatch本質(zhì)上是一個計數(shù)器,把它初始化為與執(zhí)行任務(wù)相同的數(shù)量,當一個任務(wù)執(zhí)行完時,就將計數(shù)器的值減1,直到計算器達到0時,表示完成了所有任務(wù),在await上等待線程就繼續(xù)執(zhí)行。
為上述業(yè)務(wù)場景封裝的工具類,傳入兩個參數(shù):一個參數(shù)是計算的task數(shù)量,另外一個參數(shù)是整個大任務(wù)超時的毫秒數(shù)。
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
public class ParallelCollector {
private Long timeout;
private CountDownLatch countDownLatch;
ThreadPoolExecutor executor = new ThreadPoolExecutor(100, 200, 1, TimeUnit.HOURS, new ArrayBlockingQueue<>(100));
public ParallelCollector(int taskSize, Long timeOutMill) {
countDownLatch = new CountDownLatch(taskSize);
timeout = timeOutMill;
}
public void submitTask(Runnable runnable) {
executor.execute(() -> {
runnable.run();
countDownLatch.countDown();
});
}
public void await() {
try {
this.countDownLatch.await(timeout, TimeUnit.MILLISECONDS);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public void destroy() {
this.executor.shutdown();
}
}
當任務(wù)運行時間超過了任務(wù)的時間上限,就被直接停止,這就是await()的功能。
interface是一個模擬遠程服務(wù)的超時的測試類,程序運行后,會輸出執(zhí)行結(jié)果到map集合。
public class InterfaceMock {
private volatile int num=1;
public String slowMethod1() {
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
return String.valueOf(num+1);
};
public String slowMethod2() {
return String.valueOf(num+1);
};
public String slowMethod3() {
return String.valueOf(num+1);
};
}
并行執(zhí)行獲取結(jié)果測試類
@SpringBootTest
class ThreadPoolApplicationTests {
@Test
void testTask() {
InterfaceMock interfaceMock = new InterfaceMock();
ParallelCollector collector = new ParallelCollector(3, 20L);
ConcurrentHashMap<String, String> map = new ConcurrentHashMap<>();
collector.submitTask(()->map.put("method1",interfaceMock.slowMethod1()));
collector.submitTask(()->map.put("method2",interfaceMock.slowMethod2()));
collector.submitTask(()->map.put("method3",interfaceMock.slowMethod3()));
collector.await();
System.out.println(map.toString());
collector.destroy();
}
}
當method1()執(zhí)行時間大于20ms,則該方法直接被終止,結(jié)果map集沒有method1()的結(jié)果,結(jié)果如下:
總結(jié)
使用這種方式,接口能在固定時間內(nèi)返回,注意CountDownLatch定義數(shù)量是任務(wù)個數(shù),使用concurrentHashMap避免了并行執(zhí)行時發(fā)生錯亂,造成錯誤的結(jié)果的問題。
到此這篇關(guān)于Java實現(xiàn)限定時間CountDownLatch并行場景的文章就介紹到這了,更多相關(guān)Java CountDownLatch并行場景內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
微服務(wù)架構(gòu)設(shè)計RocketMQ進階事務(wù)消息原理詳解
這篇文章主要介紹了為大家介紹了微服務(wù)架構(gòu)中RocketMQ進階層面事務(wù)消息的原理詳解,有需要的朋友可以借鑒參考下希望能夠有所幫助2021-10-10
關(guān)于Spring的@Autowired依賴注入常見錯誤的總結(jié)
有時我們會使用@Autowired自動注入,同時也存在注入到集合、數(shù)組等復雜類型的場景。這都是方便寫 bug 的場景,本篇文章帶你了解Spring @Autowired依賴注入的坑2021-09-09
Windows環(huán)境下重啟jar服務(wù)bat代碼的解決方案
在Windows環(huán)境下部署java的jar包,若有多個服務(wù)同時啟動,很難找到相應(yīng)服務(wù)重啟,每次都重啟全部服務(wù)很麻煩,應(yīng)用場景大多用于部署測試,今天給大家分享Windows環(huán)境下重啟jar服務(wù)bat代碼,感興趣的朋友一起看看吧2023-08-08
springboot?使用clickhouse實時大數(shù)據(jù)分析引擎(使用方式)
這篇文章主要介紹了springboot?使用clickhouse實時大數(shù)據(jù)分析引擎的方法,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧2022-02-02
解決springboot MultipartFile文件上傳遇到的問題
本文給大家?guī)砹私鉀Qspringboot MultipartFile文件上傳遇到的問題,解決方法超簡單,感興趣的朋友參考下本文2018-08-08
Java與Python之間使用jython工具類實現(xiàn)數(shù)據(jù)交互
今天小編就為大家分享一篇關(guān)于Java與Python之間使用jython工具類實現(xiàn)數(shù)據(jù)交互,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-03-03
Spring?Boot?3?整合?MinIO?實現(xiàn)分布式文件存儲的全過程
本文介紹了如何使用SpringBoot3和MinIO實現(xiàn)分布式文件存儲,通過MinIO的分布式對象存儲系統(tǒng),可以解決傳統(tǒng)單機文件存儲方案在面對大規(guī)模數(shù)據(jù)和高并發(fā)訪問時的不足,文章詳細講解了MinIO的安裝、配置和使用,感興趣的朋友一起看看吧2025-03-03

