亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

詳解Java中CountDownLatch異步轉(zhuǎn)同步工具類

瀏覽:62日期:2022-08-09 15:30:46
使用場(chǎng)景

由于公司業(yè)務(wù)需求,需要對(duì)接socket、MQTT等消息隊(duì)列。眾所周知 socket 是雙向通信,socket的回復(fù)是人為定義的,客戶端推送消息給服務(wù)端,服務(wù)端的回復(fù)是兩條線。無(wú)法像http請(qǐng)求有回復(fù)。下發(fā)指令給硬件時(shí),需要校驗(yàn)此次數(shù)據(jù)下發(fā)是否成功。用戶體驗(yàn)而言,點(diǎn)擊按鈕就要知道此次的下發(fā)成功或失敗。

詳解Java中CountDownLatch異步轉(zhuǎn)同步工具類

如上圖模型,

第一種方案使用Tread.sleep優(yōu)點(diǎn):占用資源小,放棄當(dāng)前cpu資源缺點(diǎn): 回復(fù)速度快,休眠時(shí)間過(guò)長(zhǎng),仍然需要等待休眠結(jié)束才能返回,響應(yīng)速度是固定的,無(wú)法及時(shí)響應(yīng)第二種方案使用CountDownLatch

package com.lzy.demo.delay;import java.util.Map;import java.util.concurrent.ArrayBlockingQueue;import java.util.concurrent.ConcurrentHashMap;import java.util.concurrent.CountDownLatch;import java.util.concurrent.DelayQueue;import java.util.concurrent.Delayed;import java.util.concurrent.ExecutorService;import java.util.concurrent.ThreadPoolExecutor;import java.util.concurrent.TimeUnit;public class CountDownLatchPool { //countDonw池 private final static Map<Integer, CountDownLatch> countDownLatchMap = new ConcurrentHashMap<>(); //延遲隊(duì)列 private final static DelayQueue<MessageDelayQueueUtil> delayQueue = new DelayQueue<>(); private volatile static boolean flag =false; //單線程池 private final static ExecutorService t = new ThreadPoolExecutor(1, 1,0L, TimeUnit.MILLISECONDS,new ArrayBlockingQueue<>(1)); public static void addCountDownLatch(Integer messageId) {CountDownLatch countDownLatch = countDownLatchMap.putIfAbsent(messageId,new CountDownLatch(1) );if(countDownLatch == null){ countDownLatch = countDownLatchMap.get(messageId);}try { addDelayQueue(messageId); countDownLatch.await(3L, TimeUnit.SECONDS);} catch (InterruptedException e) { e.printStackTrace();}System.out.println('阻塞等待結(jié)束~~~~~~'); } public static void removeCountDownLatch(Integer messageId){CountDownLatch countDownLatch = countDownLatchMap.get(messageId);if(countDownLatch == null) return;countDownLatch.countDown();countDownLatchMap.remove(messageId);System.out.println('清除Map數(shù)據(jù)'+countDownLatchMap); } private static void addDelayQueue(Integer messageId){delayQueue.add(new MessageDelayQueueUtil(messageId));clearMessageId(); } private static void clearMessageId(){synchronized (CountDownLatchPool.class){ if(flag){return; } flag = true;}t.execute(()->{ while (delayQueue.size() > 0){System.out.println('進(jìn)入線程并開(kāi)始執(zhí)行');try { MessageDelayQueueUtil take = delayQueue.take(); Integer messageId1 = take.getMessageId(); removeCountDownLatch(messageId1); System.out.println('清除隊(duì)列數(shù)據(jù)'+messageId1);} catch (InterruptedException e) { e.printStackTrace();} } flag = false; System.out.println('結(jié)束end----');}); } public static void main(String[] args) throws InterruptedException {/*測(cè)試超時(shí)清空mapnew Thread(()->addCountDownLatch(1)).start();new Thread(()->addCountDownLatch(2)).start();new Thread(()->addCountDownLatch(3)).start();*///提前創(chuàng)建線程,清空countdownnew Thread(()->{ try {Thread.sleep(500L);removeCountDownLatch(1); } catch (InterruptedException e) {e.printStackTrace(); }}).start();//開(kāi)始阻塞addCountDownLatch(1); //通過(guò)調(diào)整上面的sleep我們發(fā)現(xiàn)阻塞市場(chǎng)取決于countDownLatch.countDown()執(zhí)行時(shí)間 System.out.println('阻塞結(jié)束----'); }}class MessageDelayQueueUtil implements Delayed { private Integer messageId; private long avaibleTime; public Integer getMessageId() {return messageId; } public void setMessageId(Integer messageId) {this.messageId = messageId; } public long getAvaibleTime() {return avaibleTime; } public void setAvaibleTime(long avaibleTime) {this.avaibleTime = avaibleTime; } public MessageDelayQueueUtil(Integer messageId){this.messageId = messageId;//avaibleTime = 當(dāng)前時(shí)間+ delayTime//重試3次,每次3秒+1秒的延遲this.avaibleTime=3000*3+1000 + System.currentTimeMillis(); } @Override public long getDelay(TimeUnit unit) {long diffTime= avaibleTime- System.currentTimeMillis();return unit.convert(diffTime,TimeUnit.MILLISECONDS); } @Override public int compareTo(Delayed o) {//compareTo用在DelayedUser的排序return (int)(this.avaibleTime - ((MessageDelayQueueUtil) o).getAvaibleTime()); }}

由于socket并不確定每次都會(huì)有數(shù)據(jù)返回,所以map的數(shù)據(jù)會(huì)越來(lái)越大,最終導(dǎo)致內(nèi)存溢出需定時(shí)清除map內(nèi)的無(wú)效數(shù)據(jù)。可以使用DelayedQuene延遲隊(duì)列來(lái)處理,相當(dāng)于給對(duì)象添加一個(gè)過(guò)期時(shí)間

使用方法 addCountDownLatch 等待消息,異步回調(diào)消息清空removeCountDownLatch

到此這篇關(guān)于詳解Java中CountDownLatch異步轉(zhuǎn)同步工具類的文章就介紹到這了,更多相關(guān)CountDownLatch異步轉(zhuǎn)同步工具類內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 在线视频国产一区 | 久久色成人 | 91看片淫黄大片欧美看国产片 | 二级黄色大片 | 国产精品探花千人斩久久 | 亚洲精品小说一区二区三区 | 91麻豆传传媒波多野衣久久看 | 亚洲国产精品第一区二区三区 | 亚洲美女网站 | 91精品国产福利尤物 | 黄色毛片电影黄色毛片 | 美女内部福利视频在线观看 | 国产精品一区二区丝瓜 | 欧美一级毛片特黄大 | 国产99久久精品一区二区 | 激情丁香婷婷 | 久久久久久福利 | 久久久999久久久精品 | 欧美操片 | 中文字幕无线码中文字幕网站 | 高清毛片在线看高清 | 欧美a级v片在线观看一区 | 经典香港a毛片免费观看 | 亚洲精品国产成人中文 | 成人网址大全 | 欧美一级片在线免费观看 | 在线性视频 | 一本大道一卡2卡三卡4卡麻豆 | 一级做a爱片特黄在线观看yy | chinese国产在线视频 | 亚洲精品国产综合一线久久 | 国产精品九九视频 | 无码中文字幕乱在线观看 | 免费xxxx18美国视频 | www.看片| 欧美成人看片黄a免费看 | 在线观看国产一区二区三区99 | 精品成人资源在线观看 | a毛片免费观看 | 欧美亚洲国产精品久久蜜芽 | 国产精品人体私拍99pans |