RedisTemplate使用lua腳本分布式鎖

Jedis進行分布式鎖

Object obj = jedis.eval("if redis.call('setnx', KEYS[1], ARGV[1]) == 1 then redis.call('expire', KEYS[1], ARGV[2]) return 'true' else return 'false' end", 1, lock, val, "10");

我們使用 jedis 原生的對象執(zhí)行l(wèi)ua腳本的話居暖,非常簡單子刮,也很直觀毙沾。但是真實的項目里煌恢,我們經常會使用Spring Cloud框架骇陈,該框架已經集成了 RedisTemplate 這個類,開放了很多對redis的api瑰抵。筆者在使用RedisTemplate執(zhí)行l(wèi)ua腳本你雌,遇到一些困難。摸索了一番二汛,最后解決了婿崭,趁此機會記錄下來。

RedisTemplate執(zhí)行l(wèi)ua

/**
 * <p>
 * 分布式鎖定義
 * </p>
 *
 */
public interface DistributedLock {

    /**
     * <p>
     *     上鎖肴颊,默認的鎖的時間是 10s
     * </p>
     */
    boolean lock(String lock, String val);

    /**
     * <p>
     *     上鎖
     * </p>
     */
    boolean lock(String lock, String val, int second);

    /**
     * <p>
     *     釋放鎖
     * </p>
     */
    void unlock(String lock, String val);
}

這里氓栈,序列化的方式,采用的都是String方式婿着。實現(xiàn)類如下:

public class RedisLock implements DistributedLock {

    public static final int DEFAULT_SECOND_LEN = 10; // 10 s

    private RedisTemplate<String, String> redisTemplate;

    private static final String LOCK_LUA = "if redis.call('setnx', KEYS[1], ARGV[1]) == 1 then redis.call('expire', KEYS[1], ARGV[2]) return 'true' else return 'false' end";
    private static final String UNLOCK_LUA = "if redis.call('get', KEYS[1]) == ARGV[1] then redis.call('del', KEYS[1]) end return 'true' ";

    private RedisScript lockRedisScript;
    private RedisScript unLockRedisScript;

    private RedisSerializer<String> argsSerializer;
    private RedisSerializer<String> resultSerializer;

    /**
     * 初始化lua 腳本
     */
    public void init(RedisTemplate<String, String> redisTemplate) {

        this.redisTemplate = redisTemplate;

        argsSerializer = new StringRedisSerializer();
        resultSerializer = new StringRedisSerializer();

        lockRedisScript = RedisScript.of(LOCK_LUA, String.class);
        unLockRedisScript = RedisScript.of(UNLOCK_LUA, String.class);
    }

    @Override
    public boolean lock(String lock, String val) {
        return this.lock(lock, val, DEFAULT_SECOND_LEN);
    }

    @Override
    public boolean lock(String lock, String val, int second) {
        List<String> keys = Collections.singletonList(lock);
        String flag = redisTemplate.execute(lockRedisScript, argsSerializer, resultSerializer, keys, val, String.valueOf(second));
        return Boolean.valueOf(flag);
    }

    @Override
    public void unlock(String lock, String val) {
        List<String> keys = Collections.singletonList(lock);
        redisTemplate.execute(unLockRedisScript, argsSerializer, resultSerializer, keys, val);
    }


}

使用如下授瘦,需要先將該類初始化到Spring容器里,如下:

@Bean
public RedisLock redisLock(RedisTemplate<String, String> redisTemplate) {
   RedisLock redisLock = new RedisLock();
   redisLock.init(redisTemplate);
   return redisLock;
}

使用該鎖對象竟宋,只需要將 RedisLock 注入到所需要使用的業(yè)務類內即可提完。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市丘侠,隨后出現(xiàn)的幾起案子徒欣,更是在濱河造成了極大的恐慌,老刑警劉巖蜗字,帶你破解...
    沈念sama閱讀 221,695評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件打肝,死亡現(xiàn)場離奇詭異,居然都是意外死亡挪捕,警方通過查閱死者的電腦和手機闯睹,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,569評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來担神,“玉大人楼吃,你說我怎么就攤上這事⊥叮” “怎么了孩锡?”我有些...
    開封第一講書人閱讀 168,130評論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長亥贸。 經常有香客問我躬窜,道長,這世上最難降的妖魔是什么炕置? 我笑而不...
    開封第一講書人閱讀 59,648評論 1 297
  • 正文 為了忘掉前任荣挨,我火速辦了婚禮男韧,結果婚禮上,老公的妹妹穿的比我還像新娘默垄。我一直安慰自己此虑,他們只是感情好,可當我...
    茶點故事閱讀 68,655評論 6 397
  • 文/花漫 我一把揭開白布口锭。 她就那樣靜靜地躺著朦前,像睡著了一般。 火紅的嫁衣襯著肌膚如雪鹃操。 梳的紋絲不亂的頭發(fā)上韭寸,一...
    開封第一講書人閱讀 52,268評論 1 309
  • 那天,我揣著相機與錄音荆隘,去河邊找鬼恩伺。 笑死,一個胖子當著我的面吹牛椰拒,可吹牛的內容都是我干的莫其。 我是一名探鬼主播,決...
    沈念sama閱讀 40,835評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼耸三,長吁一口氣:“原來是場噩夢啊……” “哼乱陡!你這毒婦竟也來了?” 一聲冷哼從身側響起仪壮,我...
    開封第一講書人閱讀 39,740評論 0 276
  • 序言:老撾萬榮一對情侶失蹤憨颠,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后积锅,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體爽彤,經...
    沈念sama閱讀 46,286評論 1 318
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 38,375評論 3 340
  • 正文 我和宋清朗相戀三年缚陷,在試婚紗的時候發(fā)現(xiàn)自己被綠了适篙。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,505評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡箫爷,死狀恐怖嚷节,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情虎锚,我是刑警寧澤硫痰,帶...
    沈念sama閱讀 36,185評論 5 350
  • 正文 年R本政府宣布,位于F島的核電站窜护,受9級特大地震影響效斑,放射性物質發(fā)生泄漏。R本人自食惡果不足惜柱徙,卻給世界環(huán)境...
    茶點故事閱讀 41,873評論 3 333
  • 文/蒙蒙 一缓屠、第九天 我趴在偏房一處隱蔽的房頂上張望奇昙。 院中可真熱鬧,春花似錦敌完、人聲如沸储耐。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,357評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽弧岳。三九已至凳忙,卻和暖如春业踏,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背涧卵。 一陣腳步聲響...
    開封第一講書人閱讀 33,466評論 1 272
  • 我被黑心中介騙來泰國打工勤家, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人柳恐。 一個月前我還...
    沈念sama閱讀 48,921評論 3 376
  • 正文 我出身青樓伐脖,卻偏偏與公主長得像,于是被迫代替她去往敵國和親乐设。 傳聞我的和親對象是個殘疾皇子讼庇,可洞房花燭夜當晚...
    茶點故事閱讀 45,515評論 2 359