成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

用分布式鎖解決并發(fā)問題

Brenner / 2946人閱讀

摘要:用的方法做分布式鎖這個方案的背景主要是在和的方案上針對可能存在的死鎖問題,做了一些優(yōu)化。下面是用代碼實現(xiàn)的分布式鎖,關(guān)于部分使用的是偽代碼,請根據(jù)自己的情況用連接對象替代其中的偽代碼。

在系統(tǒng)中,當(dāng)存在多個進(jìn)程和線程可以改變某個共享數(shù)據(jù)時,就容易出現(xiàn)并發(fā)問題導(dǎo)致共享數(shù)據(jù)的不一致性。即多個進(jìn)程同時獲取到了對數(shù)據(jù)的操作權(quán)限并對數(shù)據(jù)進(jìn)行了更新,很典型的場景就是在線銷售系統(tǒng)在售賣熱銷商品時遇到多個并發(fā)請求在同一時間提交訂單的情況則極有可能造成商品超賣的現(xiàn)象。只要訪問流量不錯的系統(tǒng)都有可能遭遇并發(fā)請求造成數(shù)據(jù)庫中數(shù)據(jù)重復(fù)寫入的情況。

針對程序塊被多個進(jìn)程并發(fā)執(zhí)行問題的解決方案是確保同一個時刻同一個程序塊只能有一個進(jìn)程可執(zhí)行,其他進(jìn)程等待當(dāng)前進(jìn)程執(zhí)行完成才能獲取程序塊的執(zhí)行權(quán)對數(shù)據(jù)進(jìn)行更新,以此類推將并發(fā)執(zhí)行變?yōu)榇许樞驁?zhí)行。為了讓獲取執(zhí)行權(quán)的進(jìn)程不被其他干擾,就需要設(shè)置一個所有進(jìn)程都能讀取到的標(biāo)記,當(dāng)標(biāo)記不存在時可以設(shè)置該標(biāo)記,其余后續(xù)進(jìn)程發(fā)現(xiàn)已經(jīng)有標(biāo)記了則等待擁有標(biāo)記的進(jìn)程結(jié)束執(zhí)行程序塊取消標(biāo)記后再去嘗試設(shè)置標(biāo)記。這個標(biāo)記可以理解為鎖,設(shè)置標(biāo)記的過程就是我們通常說的加鎖。

用redis 的 setnx、expire 方法做分布式鎖

setnx()

setnx 的含義就是 SET if Not Exists,其主要有兩個參數(shù) setnx(key, value)。該方法是原子的,如果 key 不存在,則設(shè)置當(dāng)前 key 成功,返回 1;如果當(dāng)前 key 已經(jīng)存在,則設(shè)置當(dāng)前 key 失敗,返回 0。

expire()

expire 設(shè)置過期時間,要注意的是 setnx 命令不能設(shè)置 key 的超時時間,只能通過 expire() 來對 key 設(shè)置。

具體步驟

1、setnx(lockKey, 1) 如果返回 0,則說明占位失??;如果返回 1,則說明占位成功

2、expire() 命令對 lockKey 設(shè)置超時時間,為的是避免死鎖問題。

3、執(zhí)行完業(yè)務(wù)代碼后,可以通過 delete 命令刪除 key。

這個方案其實是可以解決日常工作中的需求的,但從技術(shù)方案的探討上來說,可能還有一些可以完善的地方。比如,如果在第一步 setnx 執(zhí)行成功后,在 expire() 命令執(zhí)行成功前,發(fā)生了宕機的現(xiàn)象,那么就依然會出現(xiàn)死鎖的問題,所以如果要對其進(jìn)行完善的話,可以使用 redis 的 setnx()、get() 和 getset() 方法來實現(xiàn)分布式鎖。

用 redis 的 setnx()、get()、getset()方法做分布式鎖

這個方案的背景主要是在 setnx() 和 expire() 的方案上針對可能存在的死鎖問題,做了一些優(yōu)化。

getset()

這個命令主要有兩個參數(shù) getset(key,newValue)。該方法是原子的,對 key 設(shè)置 newValue 這個值,并且返回 key 原來的舊值。假設(shè) key 原來是不存在的,那么多次執(zhí)行這個命令,會出現(xiàn)下邊的效果:

getset(key, "value1") 返回 null 此時 key 的值會被設(shè)置為 value1

getset(key, "value2") 返回 value1 此時 key 的值會被設(shè)置為 value2

依次類推!

使用步驟

setnx(lockKey, 當(dāng)前時間+過期超時時間),如果返回 1,則獲取鎖成功;如果返回 0 則沒有獲取到鎖,轉(zhuǎn)到步驟 2。

get(lockKey) 獲取值,值是當(dāng)前l(fā)ockKey的過期時間用oldExpireTime代表 ,并將這個 oldExpireTime與當(dāng)前的系統(tǒng)時間進(jìn)行比較,如果早于當(dāng)前系統(tǒng)時間,則認(rèn)為這個鎖已經(jīng)超時,可以允許別的請求重新獲取,轉(zhuǎn)向 步驟3,否則等待指定時間后返回步驟2重新開始判定。

計算 newExpireTime = 當(dāng)前時間+過期超時時間,然后 getset(lockKey, newExpireTime) 會返回當(dāng)前 lockKey 之前設(shè)置的舊值currentExpireTime。

判斷 currentExpireTime 與 oldExpireTime 是否相等,如果相等,說明當(dāng)前進(jìn)程getset 設(shè)置鎖成功,獲取到了鎖。如果不相等,說明這個鎖已經(jīng)被別的進(jìn)程獲取走了,那么當(dāng)前請求可以根據(jù)具體需求邏輯直接返回失敗,或者返回步驟2繼續(xù)重試。

在獲取到鎖之后,當(dāng)前進(jìn)程可以開始自己的業(yè)務(wù)處理,當(dāng)處理完畢后,比較當(dāng)前理時間和對鎖設(shè)置的超時時間,如果小于鎖設(shè)置的超時時間,則直接執(zhí)行 delete 釋放鎖;如果大于鎖設(shè)置的超時時間,鎖可能已由其他進(jìn)程獲得,這時執(zhí)行 delete釋放鎖的操作會導(dǎo)致把其他進(jìn)程已獲得的鎖釋放掉。

下面是用PHP代碼實現(xiàn)的Redis分布式鎖,關(guān)于Redis部分使用的是偽代碼,請根據(jù)自己的情況用Redis連接對象替代其中的偽代碼。

/**
 * 獲取Redis分布式鎖
 *
 * @param $lockKey
 * @return bool
 */
function getRedisDistributedLock(string $lockKey) : bool
{
    $lockTimeout = 2000;// 鎖的超時時間2000毫秒
    $now = intval(microtime(true) * 1000);
    $lockExpireTime = $now + $lockTimeout;
    $lockResult = Redis::setnx($lockKey, $lockExpireTime);

    if ($lockResult) {
        // 當(dāng)前進(jìn)程設(shè)置鎖成功
        return true;
    } else {
        $oldLockExpireTime = Redis::get($lockKey);
        if ($now > $oldLockExpireTime && $oldLockExpireTime == Redis::getset($lockKey, $lockExpireTime)) {
            return true;
        }
    }

    return false;
}

/**
 * 串行執(zhí)行程序
 *
 * @param string $lockKey Key for lock
 * @param Closure $closure 獲得鎖后進(jìn)程要執(zhí)行的閉包
 * @return mixed
 */
function serialProcessing(string $lockKey, Closure $closure)
{
    if (getRedisDistributedLock($lockKey)) {
        $result = $closure();
        $now = intval(microtime(true) * 1000);
        if ($now < Redis::get($lockKey)) {
            Redis::del($lockKey);   
        }
    } else {
        // 延遲200毫秒再執(zhí)行
        usleep(200 * 1000);
        return serialProcessing($lockKey, $closure);
    }

    return $result;
}

上面serialProcessing方法里當(dāng)前進(jìn)程設(shè)置鎖成功,獲取了代碼塊的執(zhí)行權(quán)后就會執(zhí)行閉包參數(shù)$closure里的代碼塊,通過傳遞閉包給方法,讓我們可以在項目任何需要確保程序串行執(zhí)行的地方使用serialProcessing方法給程序加分布式鎖解決并發(fā)請求的問題。

上面代碼實現(xiàn)用面向過程的方式是為了能簡單明了的描述怎么設(shè)置分布式鎖,讀者可以針對自己的情況執(zhí)行設(shè)計實現(xiàn)代碼。針對于大型系統(tǒng)使用集群Redis的情況,設(shè)置分布式鎖的步驟更復(fù)雜,有興趣的可以查看Redlock 算法和redissonredis分布式鎖組件。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/31005.html

相關(guān)文章

  • 布式冪等問題解決方案三部曲

    摘要:解決冪等問題的三部曲,也是作者的思考框架。這是解決冪等問題的第二部曲列出并減少副作用的分析維度。所以在并發(fā)執(zhí)行的維度,將并發(fā)重復(fù)執(zhí)行變成串行重復(fù)執(zhí)行是最好的冪等解決方案。 綱要 文章目的:本文旨在提煉一套分布式冪等問題的思考框架,而非解決某個具體的分布式冪等問題。在這個框架體系內(nèi),會有一些方案舉例說明。文章目標(biāo):希望讀者能通過這套思考框架設(shè)計出符合自己業(yè)務(wù)的完備的冪等解決方案。文章內(nèi)容...

    mumumu 評論0 收藏0
  • Redis布式解決搶購問題

    摘要:廢話不多說,首先分享一個業(yè)務(wù)場景搶購。下面就是分布式鎖的解決方法。首先要加入的依賴,該類只有兩個功能,加鎖和解鎖,解鎖比較簡單,就是刪除當(dāng)前的鍵值對。這時繼續(xù)執(zhí)行,由于所以該線程獲取到鎖。 廢話不多說,首先分享一個業(yè)務(wù)場景-搶購。一個典型的高并發(fā)問題,所需的最關(guān)鍵字段就是庫存,在高并發(fā)的情況下每次都去數(shù)據(jù)庫查詢顯然是不合適的,因此把庫存信息存入Redis中,利用redis的鎖機制來控制...

    taoszu 評論0 收藏0
  • Akka系列(六):Actor解決了什么問題?

    摘要:原文鏈接解決了什么問題使用模型來克服傳統(tǒng)面向?qū)ο缶幊棠P偷木窒扌?,并?yīng)對高并發(fā)分布式系統(tǒng)所帶來的挑戰(zhàn)。在某些情況,這個問題可能會變得更糟糕,工作線程發(fā)生了錯誤但是其自身卻無法恢復(fù)。 這段時間由于忙畢業(yè)前前后后的事情,拖更了很久,表示非常抱歉,回歸后的第一篇文章主要是看到了Akka最新文檔中寫的What problems does the actor model solve?,閱讀完后覺...

    Carson 評論0 收藏0
  • 淺談Java并發(fā)編程系列(一)—— 如何保證線程安全

    摘要:比如需要用多線程或分布式集群統(tǒng)計一堆用戶的相關(guān)統(tǒng)計值,由于用戶的統(tǒng)計值是共享數(shù)據(jù),因此需要保證線程安全。如果類是無狀態(tài)的,那它永遠(yuǎn)是線程安全的。參考探索并發(fā)編程二寫線程安全的代碼 線程安全類 保證類線程安全的措施: 不共享線程間的變量; 設(shè)置屬性變量為不可變變量; 每個共享的可變變量都使用一個確定的鎖保護(hù); 保證線程安全的思路: 1. 通過架構(gòu)設(shè)計 通過上層的架構(gòu)設(shè)計和業(yè)務(wù)分析來避...

    mylxsw 評論0 收藏0
  • 為Java程序員金三銀四精心挑選的300余道Java面試題與答案

    摘要:為程序員金三銀四精心挑選的余道面試題與答案,歡迎大家向我推薦你在面試過程中遇到的問題我會把大家推薦的問題添加到下面的常用面試題清單中供大家參考。 為Java程序員金三銀四精心挑選的300余道Java面試題與答案,歡迎大家向我推薦你在面試過程中遇到的問題,我會把大家推薦的問題添加到下面的常用面試題清單中供大家參考。 前兩天寫的以下博客,大家比較認(rèn)可,熱度不錯,希望可以幫到準(zhǔn)備或者正在參加...

    tomorrowwu 評論0 收藏0

發(fā)表評論

0條評論

Brenner

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<