Redlock實(shí)踐--非阻塞模式

本文主要介紹在琴房預(yù)約項(xiàng)目中所用到的資源訪問加鎖技術(shù)Redlock档桃,以及如果實(shí)現(xiàn)將redlock轉(zhuǎn)為非阻塞鎖粘拾。

1. 背景簡介

1.1 項(xiàng)目簡介

本次實(shí)踐項(xiàng)目是THU琴房預(yù)約小程序與web管理端,后端使用koa框架偎箫。主要功能有預(yù)約琴房(用戶)木柬、更改琴房可用時(shí)間(管理端)等。后端使用mysql存儲琴房信息(包含琴房可用時(shí)間串)淹办。

1.2 問題介紹

  • 多個(gè)用戶可能在同一時(shí)間對同一琴房的同一時(shí)間段進(jìn)行預(yù)約
  • 用戶與管理員可能同時(shí)進(jìn)行預(yù)約弄诲、更改琴房可用時(shí)間

根據(jù)上篇文章所描述的,因?yàn)榇嬖诓煌僮髦g的資源競爭,所以我們這里使用增加信號量的方式——Redlock齐遵。

2. 加鎖實(shí)踐

redlock是一個(gè)基于redis數(shù)據(jù)庫的分布式鎖寂玲。通過在redis數(shù)據(jù)庫中設(shè)定鍵值來進(jìn)行信號量的定義。

  • 我們把“預(yù)約“和“更改琴房可用時(shí)間”定義為使用同一個(gè)鍵值梗摇。
  • 在操作前首先獲取此鍵值拓哟,如果獲取不到,說明被占用伶授。

關(guān)于這個(gè)流程断序,包括set鍵值與”占用"和"未占用“狀態(tài)的更改是由Redlock實(shí)現(xiàn)的。

2.1 給資源加鎖

// 獲取權(quán)限
let redis = require("redis");
let client = redis.createClient("to change: your redis port","to change: your server ip");
let redlock = new Redlock([client]);
// 設(shè)置時(shí)間
let totalTime = 5000;
let key = "to change: as you like";
// 加鎖
redlock.lock(key, totalTime).then(async function(lock){
        // to do
        // your operation block
        // release lock
        lock.unlock().catch(function(err){})
    }
}).catch(()=>{}) // 如果不加catch會直接報(bào)錯(cuò)終止程序

主要流程為:

  • 獲取redis的操作權(quán)限
  • 設(shè)定鍵值糜烹,最長等待時(shí)間(防止一直被占用)
  • 執(zhí)行操作
  • 釋放鎖

按照上面流程违诗,我們成功進(jìn)行了加鎖。

鍵值的設(shè)置非常自由疮蹦,我們可以通過鍵值的設(shè)置诸迟,控制加鎖覆蓋的范圍以及力度。

2.2 阻塞鎖與非阻塞鎖

在完成加鎖之后愕乎,我們進(jìn)行壓力測試阵苇,發(fā)現(xiàn)了一個(gè)非常坑的情況:Redlock是非阻塞鎖感论。

這就意味著绅项,當(dāng)一個(gè)用戶進(jìn)行預(yù)約的時(shí)候,別的用戶如果有請求就會直接失敗比肄,這樣是非常用戶不友好的快耿。所以我們需要一些操作來將redlock轉(zhuǎn)為非阻塞鎖。有兩個(gè)方案:

  • 方案一:實(shí)現(xiàn)一個(gè)請求隊(duì)列以及回調(diào)函數(shù)芳绩,在資源被釋放的時(shí)候掀亥,進(jìn)行回調(diào)。
  • 方案二:設(shè)置最長等待時(shí)間示括,在此時(shí)間段內(nèi)進(jìn)行輪詢,如果超過此時(shí)間痢畜,放棄請求垛膝。

對于方案一,請求隊(duì)列可能會很長丁稀,如果前面的資源不釋放吼拥,就會一直等待;對于方案二线衫,輪詢需要消耗更多的資源凿可。

我們這里說明方案二的實(shí)現(xiàn)(目前最常用的轉(zhuǎn)阻塞鎖的方案)

let redis = require("redis");
let client = redis.createClient(config.redisPort,config.serverIp);
let redlock = new Redlock([client]);
let totalTime = 5000;
let key = "to change: as you like";
let intervalTime = 50;  // 輪詢時(shí)間間隔

let sleep = function(ms) {
    return new Promise(resolve => setTimeout(resolve, ms))
}

let tag = 0;
for(let j = 0; j<200; j++){
    redlock.lock(key, totalTime).then(async function(lock){
        if(tag === 1){
            lock.unlock().catch(function(err){})
        }
        else{
            tag = 1
            // to do
            // your operation block
            // release lock
            lock.unlock().catch(function(err){})
        }
    }).catch(()=>{})
    if(tag === 1){
        break
    }
    await sleep(intervalTime) // 設(shè)置間隔時(shí)間
}
if(tag === 0){
    errorMsg = "請求超時(shí)"
    return ;
}

3. 效果檢驗(yàn)

加鎖效果

在預(yù)約與檢票單項(xiàng)測試的時(shí)候,1000個(gè)人中只有一個(gè)人成功,符合預(yù)期枯跑。

加鎖效果

在預(yù)約與更改琴房可用時(shí)間混合測試的時(shí)候惨驶,2000個(gè)人中只有一個(gè)人成功,符合預(yù)期敛助。

以上粗卜,說明這樣加鎖,可以成功解決資源訪問競爭情況纳击。

4. 參考資料

  1. redlock 效果解析

  2. redlock git地址

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末续扔,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子焕数,更是在濱河造成了極大的恐慌纱昧,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,029評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件堡赔,死亡現(xiàn)場離奇詭異识脆,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)加匈,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,395評論 3 385
  • 文/潘曉璐 我一進(jìn)店門存璃,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人雕拼,你說我怎么就攤上這事纵东。” “怎么了啥寇?”我有些...
    開封第一講書人閱讀 157,570評論 0 348
  • 文/不壞的土叔 我叫張陵偎球,是天一觀的道長。 經(jīng)常有香客問我辑甜,道長衰絮,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,535評論 1 284
  • 正文 為了忘掉前任磷醋,我火速辦了婚禮猫牡,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘邓线。我一直安慰自己淌友,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,650評論 6 386
  • 文/花漫 我一把揭開白布骇陈。 她就那樣靜靜地躺著震庭,像睡著了一般。 火紅的嫁衣襯著肌膚如雪你雌。 梳的紋絲不亂的頭發(fā)上器联,一...
    開封第一講書人閱讀 49,850評論 1 290
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼拨拓。 笑死肴颊,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的千元。 我是一名探鬼主播苫昌,決...
    沈念sama閱讀 39,006評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼幸海!你這毒婦竟也來了祟身?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,747評論 0 268
  • 序言:老撾萬榮一對情侶失蹤物独,失蹤者是張志新(化名)和其女友劉穎袜硫,沒想到半個(gè)月后握牧,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體喜最,經(jīng)...
    沈念sama閱讀 44,207評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,536評論 2 327
  • 正文 我和宋清朗相戀三年咧纠,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了官研。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片秽澳。...
    茶點(diǎn)故事閱讀 38,683評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖戏羽,靈堂內(nèi)的尸體忽然破棺而出担神,到底是詐尸還是另有隱情,我是刑警寧澤始花,帶...
    沈念sama閱讀 34,342評論 4 330
  • 正文 年R本政府宣布妄讯,位于F島的核電站,受9級特大地震影響酷宵,放射性物質(zhì)發(fā)生泄漏亥贸。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,964評論 3 315
  • 文/蒙蒙 一浇垦、第九天 我趴在偏房一處隱蔽的房頂上張望炕置。 院中可真熱鬧,春花似錦男韧、人聲如沸朴摊。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,772評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽仍劈。三九已至厕倍,卻和暖如春寡壮,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,004評論 1 266
  • 我被黑心中介騙來泰國打工况既, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留这溅,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,401評論 2 360
  • 正文 我出身青樓棒仍,卻偏偏與公主長得像悲靴,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子莫其,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,566評論 2 349

推薦閱讀更多精彩內(nèi)容