基於Mongodb分散式鎖簡單實現,解決定時任務併發執行問題

来源:https://www.cnblogs.com/surging-dandelion/archive/2023/04/18/17330389.html
-Advertisement-
Play Games

前言 我們日常開發過程,會有一些定時任務的代碼來統計一些系統運行數據,但是我們應用有需要部署多個實例,傳統的通過配置文件來控制定時任務是否啟動又太過繁瑣,而且還經常出錯,導致一些異常數據的產生 網上有很多分散式鎖的實現方案,基於redis、zk、等有很多,但是我的就是一個用了mysql和mongo的 ...


前言

我們日常開發過程,會有一些定時任務的代碼來統計一些系統運行數據,但是我們應用有需要部署多個實例,傳統的通過配置文件來控制定時任務是否啟動又太過繁瑣,而且還經常出錯,導致一些異常數據的產生

網上有很多分散式鎖的實現方案,基於redis、zk、等有很多,但是我的就是一個用了mysql和mongo的小應用,不准備引入其他三方中間件來解決這個問題,擼一個簡單的分散式鎖來解決定時任務併發執行的問題,加鎖操作的原子性和防死鎖也都要支持,這裡我使用mongodb寫了AllInOne的工具類

All in one Code

先上代碼

@Component
@Slf4j
public class MongoDBLock {

    private static final int DEFAULT_LOCK_TIMEOUT = 30;//鎖的預設超時時間,單位秒

    private MongoTemplate mongoTemplate;
    private int lockTimeout;

    public MongoDBLock(MongoTemplate mongoTemplate) {
        this.mongoTemplate = mongoTemplate;
        this.lockTimeout = DEFAULT_LOCK_TIMEOUT;
    }

    /**
     * 嘗試獲取分散式鎖
     *
     * @param lockKey 鎖的key
     * @return true:獲取鎖成功,false:獲取鎖失敗
     */
    private boolean acquireLock(String lockKey) {
        LockDocument document = new LockDocument();
        document.setId(lockKey);
        document.setExpireAt(Instant.ofEpochMilli(Instant.now().toEpochMilli() + lockTimeout * 1000));
        try {
            mongoTemplate.insert(document);
            return true;
        } catch (Exception e) {

        }
        return false;
    }

    /**
     * 釋放分散式鎖
     *
     * @param lockKey 鎖的key
     */
    private void releaseLock(String lockKey) {
        Query query = new Query(Criteria.where("key").is(lockKey));
        mongoTemplate.remove(query, LockDocument.class);
        log.info("程式執行成功,釋放分散式鎖,lockKey:{}",lockKey);
    }

    /**
     * 分散式鎖入口方法,參數lockName為鎖的名稱,lockKey為需要加鎖的key,執行完成後自動釋放鎖
     *
     * @param lockKey
     * @param task
     * @param <T>
     * @throws Exception
     */
    public <T> void executeWithLock(String lockKey, ITask<T> task) throws Exception {
        boolean locked = acquireLock(lockKey);
        if (locked) {
            log.info("獲取分散式鎖成功,lockKey:{}",lockKey);
            try {
                task.execute();
            } finally {
                releaseLock(lockKey);
            }
        } else {
            log.warn("獲取分散式鎖失敗,lockKey:{}", lockKey);
            throw new AppException("獲取分散式鎖失敗!");
        }
    }

    @Data
    @Document(collection = "lock_collection")
    static class LockDocument {
        @Id
        private String id;
        @Indexed(expireAfterSeconds = DEFAULT_LOCK_TIMEOUT)
        private Instant expireAt;
    }

    @FunctionalInterface
    public interface ITask<T> {
        T execute() throws Exception;
    }
}

調用示例

    @Resource
    MongoDBLock mongoDBLock;

    mongoDBLock.executeWithLock("key", () -> {
        // do some thing
        return null;
    });

原理

  • 使用key作為主鍵,利用mongodb的insert原子性保障LockDocument不會重覆插入
  • LockDocument中expireAt欄位利用的mongodb索引過期機制,解決死鎖問題,這裡設置超時時間是30秒,併在執行完成之後會主動釋放鎖

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本文首發於公眾號:Hunter後端 原文鏈接:Django筆記二十四之資料庫函數之比較和轉換函數 這一篇筆記開始介紹幾種資料庫函數,以下是幾種函數及其作用 Cast 轉換類型 Coalesce 優先取值 Greatest 返回較大值 Nullif 值相同返回 None 1、model 準備 這一篇筆 ...
  • 網路分層結構 電腦網路體系大致分為三種,OSI七層模型、TCP/IP四層模型和五層模型。一般面試的時候考察比較多的是五層模型。最全面的Java面試網站 五層模型:應用層、傳輸層、網路層、數據鏈路層、物理層。 應用層:為應用程式提供交互服務。在互聯網中的應用層協議很多,如功能變數名稱系統DNS、HTTP協議 ...
  • 一、哆啦A夢 由於代碼過長,這裡僅顯示部分代碼: from turtle import * import turtle as t from random import * #五軌跡跳躍 def my_goto(x,y): penup() goto(x,y) pendown() def eyes(): ...
  • Go語言流媒體開源項目 LAL 今天發佈了v0.35.4版本。 LAL 項目地址:https://github.com/q191201771/lal 老規矩,簡單介紹一下: ▦ 一. OBS支持RTMP H265推流 新出的標準,一般被稱為enhanced RTMP,OBS新版(29.1+版本,點我 ...
  • 什麼是冒泡排序 冒泡排序(Bubble Sort)也是一種簡單直觀的排序演算法。它重覆地走訪過要排序的數列,一次比較兩個元素,如果他們的順序錯誤就把他們交換過來。走訪數列的工作是重覆地進行直到沒有再需要交換,也就是說該數列已經排序完成。這個演算法的名字由來是因為越小的元素會經由交換慢慢"浮"到數列的頂端 ...
  • 平時開發時的工作的話之主要負責寫代碼就行了,什麼發佈項目啊,好吧不是我們乾的事。在我們的瞭解中打包發佈項目應該不是一個困難的問題。 對,最簡單的方法就行使用直接使用maven插件打包,甚至我們都不需要知道他是怎麼實現的,插件能幫我們將項目打包為一個jar包,然後使用java -jar xx.jar就 ...
  • 使用第三方jar包,完成get/set操作 Lombok,結合特殊的註解,實現setter和getter的自動生成 導入jar包 使用插件Lombok 在類里import 即可使用 import lombok.AllArgsConstructor; import lombok.Data; impor ...
  • 本文介紹基於Python中ArcPy模塊,對大量長時間序列柵格遙感影像文件的每一個像元進行多時序平均值的求取。 在遙感應用中,我們經常需要對某一景遙感影像中的全部像元的像素值進行平均值求取——這一操作很好實現,基於ArcMap軟體或者簡單的Python代碼就可以實現;但有時候,我們會需要結合同一地區 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...