針對RedisTemplate分布式鎖實現WatchDog

2023-04-17 08:45:34 來源:騰訊云


(資料圖片)

在此之前,去看了下Redission的實現原理,不過在開發中,原本的代碼使用RedistTemplate實現的,也不太想換,所以我想了下,不如自己實現要給WatchDog。

我的想法是,在用戶加上鎖的時候開啟個定時任務線程,并且在定時任務中,判斷原線程isAlive狀態進行“續命”。

下面是代碼(在這里面為了方便,未使用的是HuTool.CornUtil來實現動態定時任務):

/** * Title * * @ClassName: LockUtil * @Description:鎖工具類,通過內部枚舉類實現單例,防止反射攻擊 * @author: Karos * @date: 2023/1/4 0:17 * @Blog: https://www.wzl1.top/ */package cn.katool.lock;import cn.hutool.core.util.BooleanUtil;import cn.hutool.core.util.ObjectUtil;import cn.hutool.cron.CronUtil;import cn.hutool.cron.task.Task;import cn.katool.Config.LockConfig;import cn.katool.Exception.ErrorCode;import cn.katool.Exception.KaToolException;import cn.katool.other.MethodIntefaceUtil;import com.qiniu.util.StringUtils;import lombok.SneakyThrows;import lombok.extern.slf4j.Slf4j;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Scope;import org.springframework.data.redis.core.RedisTemplate;import org.springframework.stereotype.Component;import org.springframework.util.ObjectUtils;import javax.annotation.Resource;import java.util.concurrent.ConcurrentHashMap;import java.util.concurrent.TimeUnit;@Component@Scope("prototype")@Slf4jpublic class LockUtil {        @Resource        RedisTemplate redisTemplate;        private LockUtil(){        }        private static boolean isOpenCorn=false;        /**         * 帶看門狗機制上鎖         * @param lockObj         * @return         */        public boolean DistributedLock(Object lockObj){                try {                        return DistributedLock(lockObj,null,null);                } catch (KaToolException e) {                        throw new RuntimeException(e);                }        }        @Resource        LockConfig lockConfig;        //加鎖        /**         * 無看門狗機制上鎖         * @param obj         * @param exptime         * @param timeUnit         * @return         * @throws KaToolException         */        public boolean DistributedLock(Object obj,Long exptime,TimeUnit timeUnit) throws KaToolException {                if (ObjectUtil.isEmpty(obj)){                        throw new KaToolException(ErrorCode.PARAMS_ERROR," Lock=> 傳入obj為空");                }                Boolean isDelay=false;                if (ObjectUtil.isAllEmpty(exptime,timeUnit)){                        isDelay=true;                }                if(ObjectUtil.isEmpty(exptime)){                        exptime= lockConfig.getInternalLockLeaseTime();;                }                if (ObjectUtils.isEmpty(timeUnit)){                        timeUnit=lockConfig.getTimeUnit();                }                //線程被鎖住了,就一直等待                DistributedAssert(obj);                Boolean aBoolean = redisTemplate.opsForValue().setIfAbsent("Lock:"+obj.toString(), "1", exptime, timeUnit);                log.info("katool=> LockUntil => DistributedLock:{} value:{} extime:{} timeUnit:{}",obj.toString(), "1", exptime, timeUnit);                //實現看門狗                if (isDelay){                        if (LockUtil.isOpenCorn==false){                                //如果同一個項目之前打開過,那么先關閉,避免重復啟動                                CronUtil.stop();                                //支持秒級別定時任務                                CronUtil.setMatchSecond(true);                                //定時服務啟動                                CronUtil.start();                                LockUtil.isOpenCorn=true;                        }                        Thread thread = Thread.currentThread();                        TimeUnit finalTimeUnit = timeUnit;                        Long finalExptime = exptime;                        class TempClass{                                public String scheduleId;                        }                        final TempClass tempClass = new TempClass();                        tempClass.scheduleId=CronUtil.schedule("0/30 * * * * ?", new Task() {                                @SneakyThrows                                @Override                                public void execute() {                                        boolean alive = thread.isAlive();                                        if (alive) {                                                delayDistributedLock(obj, finalExptime>=3?(finalExptime / 3):finalExptime, finalTimeUnit);                                                return;                                        } else {                                                if (tempClass.scheduleId==null||"".equals(tempClass.scheduleId)){                                                        return;                                                }                                                CronUtil.remove(tempClass.scheduleId);                                                DistributedUnLock(obj);                                                return;                                        }                                }                        });                }                return BooleanUtil.isTrue(aBoolean);        }        //檢鎖        public void DistributedAssert(Object obj) throws KaToolException {                if (ObjectUtils.isEmpty(obj)){                        throw new KaToolException(ErrorCode.PARAMS_ERROR," Lock=> 傳入obj為空");                }                while(true){                        Object o = redisTemplate.opsForValue().get("Lock:" + obj.toString());                        if (ObjectUtils.isEmpty(o))return;                }        }        //延期        public boolean delayDistributedLock(Object obj,Long exptime,TimeUnit timeUnit) throws KaToolException {                if (ObjectUtils.isEmpty(obj)){                        throw new KaToolException(ErrorCode.PARAMS_ERROR," Lock=> 傳入obj為空");                }                Boolean aBoolean = redisTemplate.opsForValue().setIfPresent("Lock:"+obj.toString(), "1", exptime, timeUnit);                log.info("katool=> LockUntil => delayDistributedLock:{} value:{} extime:{} timeUnit:{}",obj.toString(), "1", exptime, timeUnit);                return BooleanUtil.isTrue(aBoolean);        }        //釋放鎖        public boolean DistributedUnLock(Object obj) throws KaToolException {                if (ObjectUtils.isEmpty(obj)){                        throw new KaToolException(ErrorCode.PARAMS_ERROR," Lock=> 傳入obj為空");                }                Boolean aBoolean = redisTemplate.delete("Lock:" + obj.toString());                log.info("katool=> LockUntil => unDistributedLock:{} isdelete:{} ",obj.toString(),true);                return BooleanUtil.isTrue(aBoolean);        }        //利用枚舉類實現單例模式,枚舉類屬性為靜態的        private enum SingletonFactory{                Singleton;                LockUtil lockUtil;                private SingletonFactory(){                        lockUtil=new LockUtil();                }                public LockUtil getInstance(){                        return lockUtil;                }        }        @Bean("LockUtil")        public static LockUtil getInstance(){                return SingletonFactory.Singleton.lockUtil;        }}

關鍵詞:

相關文章

熱文推薦

針對RedisTemplate分布式鎖實現WatchDog
針對RedisTemplate分布式鎖實現WatchDog

在此之前,去看了下Redission的實現原理,不過在開發......更多>

早知道:上市公司獨立董事制度全面改革;寧德時代鈉離子電池首發落地
早知道:上市公司獨立董事制度全面改革;寧德時代鈉離子電池首發落地

上市公司獨立董事制度全面改革,履職更具“獨立性”;......更多>

廣發證券:AI投資會有一段時間調整期,下一階段依靠兩方面外界因素_全球快資訊
廣發證券:AI投資會有一段時間調整期,下一階段依靠兩方面外界因素_全球快資訊

廣發證券最新研報表示,AI方面的投資會有一段時間的調......更多>

環球動態:華泰證券:關注AI大模型在終端側的輕量化應用
環球動態:華泰證券:關注AI大模型在終端側的輕量化應用

華泰證券研報表示,模型壓縮技術是實現AI大模型在邊 ......更多>

排行推薦

旺能環境:簽署《監利市餐廚垃圾綜合處理和資源化利用項目特許經營協議》|世界簡訊
旺能環境:簽署《監利市餐廚垃圾綜合處理和資源化利用項目特許經營協議》|世界簡訊
旺能環境(002034):簽署《監利市餐廚垃圾綜合處理和資... 更多>
環球微頭條丨中證報:因城施策持續顯效,樓市企穩跡象漸增
環球微頭條丨中證報:因城施策持續顯效,樓市企穩跡象漸增
中指研究院認為,各地因城施策支持力度不斷增強,短期... 更多>
證券日報:旅游市場加快復蘇, “五一”出行熱度高漲
證券日報:旅游市場加快復蘇, “五一”出行熱度高漲
攜程旗下FlightAi市場洞察平臺數據顯示,截至4月14日... 更多>
中信建投:光伏估值處于低位 看好一季報業績超預期方向_世界即時
中信建投:光伏估值處于低位 看好一季報業績超預期方向_世界即時
中信建投最新研報表示,市場對于光伏板塊預期悲觀,主... 更多>
世界今日報丨俄羅斯總統普京會見李尚福
據新華社,當地時間4月16日,俄羅斯總統普京在莫斯科... 更多>
全球今熱點:經濟日報:確保電力平穩迎峰度夏
文章稱,隨著社會生產生活快速恢復,今年迎峰度夏期間... 更多>
經濟日報:多渠道拓展國產大豆消費|全球滾動
國家糧食和物資儲備局科學研究院首席研究員李愛科撰文... 更多>
天天熱訊:國內成品油零售價將迎今年來最大漲幅,92號汽油每升上漲0.4元左右
受國際油價大幅攀升影響,國內汽柴油零售價結束連跌走... 更多>
【環球新視野】產業趨勢投資中,如何賺“認知差”的錢?
投資進化論系列:消除不確定性這一輪令人瞠目結舌的AI... 更多>
全球觀熱點:寫給所有散戶的一封信:換手率大于7%意味著什么?看懂少走彎路!
所謂市場,就是交易場所。市場本身不產生利潤。所謂賺... 更多>
上海誼眾紫杉醇聚合物膠束新增適應癥臨床獲批
$上海誼眾(SH688091)$從國家藥品監督管理局藥品評審中... 更多>
機構開始覆蓋光庫_觀點
光庫周五漲13 69%。機構覆蓋開始了。海通吹響號角,... 更多>
30年期國債期貨即將上市 首批3個合約將于4月21日交易
具體而言,中金所在《通知》中稱,30年期國債期貨首批... 更多>
時訊:好奇和天真
$東阿阿膠(SZ000423)$自己的簽名檔從來到雪球就是這個... 更多>

世界動態:今年應該買什么(主線順序

淺談華晨中國01114的估值和安全邊際

【價值重估】醫藥板塊逆勢上漲,AI

滾動:無名最終票房具體詳細內容是

當前資訊!人頭攢動 掌聲連連!廣交

世界新消息丨持有興業銀行的投資者

埃安肖勇:特斯拉不是不能戰勝

寧德時代鈉離子電池首發落地奇瑞車

繼續討論不同輪動日的影響

4月16日:|世界快資訊

九九在线精品视频专区| 亚洲日韩国产AV无码无码精品| 国产精品无码AV天天爽播放器 | 精品400部自拍视频在线播放| 香蕉久久夜色精品升级完成| 国产成人无码精品一区不卡| 在线精品视频播放| 少妇人妻偷人精品一区二区| 国产精品亚洲二区在线| 精品一区二区三区免费毛片爱| 国产精品成人观看视频| 国产精品国产AV片国产| 999精品视频在线观看热6| 精品国产亚洲男女在线线电影 | 日韩精品系列产品| 久久久久久影院久久久久免费精品国产小说| 久久精品国产亚洲Aⅴ香蕉| 6一12呦女精品| 精品91自产拍在线观看二区| 国产色婷婷精品综合在线| 精品视频一区二区三区四区五区| 亚洲午夜精品久久久久久浪潮 | 99久热re在线精品996热视频| 无码人妻精品中文字幕| 亚洲AV成人无码久久精品老人| 97视频精品视频人人一二区| 一本一道久久a久久精品综合 | 久久精品视频久久| 精品国产毛片一区二区无码| 国产高清在线精品二区| 国产精品国产三级国产专播| 99热这里只有精品7| 真实国产乱子伦精品视频| 国产精品美女久久久网AV| 国产一区精品视频| 日韩人妻高清精品专区| 欧美成人精品三级网站下载 | 久久精品国产亚洲AV无码偷窥| 久久精品一区二区三区不卡| 国内精品伊人久久久久网站| 亚洲国产一成久久精品国产成人综合|