redis setnx 鎖在單機(jī)架構(gòu)中使用示
應(yīng)用場(chǎng)景
- 在應(yīng)用系統(tǒng)中,在涉及對(duì)數(shù)據(jù)庫(kù)增刪改的情況下当犯,即使前端做了相關(guān)限制休玩,還是有可能因?yàn)榫W(wǎng)絡(luò)或異常操作產(chǎn)生并發(fā)重復(fù)調(diào)用的情況,導(dǎo)致對(duì)相通記錄進(jìn)行重復(fù)處理
- 高并發(fā)情況下菇存,在請(qǐng)求的接口中對(duì)數(shù)據(jù)有增刪改的情況下
3.為了防止這種情況對(duì)系統(tǒng)造成異常影響,我們通過(guò)Redis實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的鎖邦蜜,每個(gè)請(qǐng)求需先獲取鎖才能執(zhí)行業(yè)務(wù)邏輯依鸥,執(zhí)行結(jié)束后才會(huì)釋放鎖;保證了同一數(shù)據(jù)的并發(fā)重復(fù)操作請(qǐng)求只有一個(gè)請(qǐng)求可以獲取到鎖(依賴Redis的單線程)
場(chǎng)景示例
模擬簡(jiǎn)單的下單減庫(kù)存的場(chǎng)景悼沈,使用ab壓力測(cè)試模擬并發(fā)請(qǐng)求贱迟。
- 數(shù)據(jù)庫(kù)中g(shù)ood_id =1 商品剩余庫(kù)存 10
- 模擬100次請(qǐng)求 ,每次并發(fā) 10個(gè)請(qǐng)求
# 模擬50次請(qǐng)求 ,每次并發(fā) 5個(gè)請(qǐng)求
ab -n 50 -c 5 127.0.0.1:80/redis-lock/index.php
在代碼中將$flag設(shè)置為false 如 圖-05,不使用redis setnx , 模擬并發(fā)請(qǐng)求前庫(kù)存剩余10件 如: 圖-01 所示,模擬并發(fā)請(qǐng)求后,庫(kù)存剩余 0件 如:圖-02 所示, 生成的用戶定單 如:圖-03所示,此時(shí)下單記錄總商品數(shù)為13 如:圖-04 所示, 下單數(shù) 13 大于 庫(kù)存數(shù)據(jù)10絮供,并發(fā)情況下出現(xiàn)超買(mǎi)的情況
使用 setnx
# 開(kāi)啟redis的setnx 使用redis鎖,進(jìn)行模擬并發(fā)請(qǐng)求
ab -n 50 -c 5 127.0.0.1:80/redis-lock/index.php
如圖-06 所示在代碼中將 $flag 設(shè)置為 true衣吠,開(kāi)啟redis 的setnx,模擬并發(fā)請(qǐng)求,如圖-07 ,圖-08 所示,沒(méi)有出現(xiàn)超買(mǎi)的情況
核心代碼示例:
如圖-10 核心代碼示例壤靶,是進(jìn)行獲取鎖,設(shè)置鎖,和解鎖
如圖-09 所示 是在業(yè)務(wù)處理開(kāi)始首先進(jìn)行鎖的判斷處理缚俏,當(dāng)前進(jìn)程只有獲取鎖成功才能進(jìn)行業(yè)務(wù)處理,否則將終止程序的運(yùn)行