精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

手寫一個Redis分布式鎖,讓你徹底搞懂

數據庫 Redis
如果程序運行的極慢(硬件處理慢或者進行了GC),導致30秒已經到了,鎖已經失效了,程序還沒有運行完成,這時候,就會有另一個線程總想鉆個空子,導致票的超賣問題。

哈嘍,大家好,我是指北君。

今天帶大家深入剖析一下Redis分布式鎖,徹底搞懂它。

場景

既然要搞懂Redis分布式鎖,那肯定要有一個需要它的場景。

高并發售票問題就是一個經典案例。

搭建環境

  • 準備redis服務,設置redis的鍵值對:set ticket 10
  • 準備 postman、JMeter 等模擬高并發請求的工具
  • 核心代碼
@Service
public class TicketServiceImpl implements TicketService {
@Autowired
private StringRedisTemplate stringRedisTemplate;

private Logger logger = LoggerFactory.getLogger(TicketServiceImpl.class);

@Override
public String sellTicket(){
String ticketStr = stringRedisTemplate.opsForValue().get("ticket");
int ticket = 0;
if (null != ticketStr) {
ticket = Integer.parseInt(ticketStr);
}
if (ticket > 0) {
int ticketNew = ticket - 1;
stringRedisTemplate.opsForValue().set("ticket", String.valueOf(ticketNew));
logger.info("當前票的庫存為:" + ticketNew);
} else {
logger.info("手速不夠呀,票已經賣光了...");
}
return "搶票成功...";
}
}

分析解決問題

以上代碼沒有做任何的加鎖操作,在高并發情況下,票的超賣情況很嚴重,根本無法正常使用

分析1

既然要加分布式鎖,那么我們可以使用Redis中的setnx命令來模擬一個鎖。

redis> EXISTS job                # job 不存在
(integer) 0

redis> SETNX job "programmer" # job 設置成功
(integer) 1

redis> SETNX job "code-farmer" # 嘗試覆蓋 job ,失敗
(integer) 0

當一個線程進入到當前方法中,使用 setnx 設置一個鍵,如果設置成功,就允許繼續訪問,設置失敗,就不能訪問該方法;

當方法運行完畢時,將這個鍵刪除,下一次再有線程來訪問時,就重新執行該操作。

public String sellTicket(){
String lock="lock";
// 如果成功設置這個值,證明目前該方法并沒有被操作,可以進行賣票操作
Boolean tag = stringRedisTemplate.opsForValue().setIfAbsent(lock, "");
if (!tag) { // 如果設置失敗,證明當前方法正在被執行,不允許再次執行
// 實際開發環境應該使用隊列來完成訪問操作,這里主要探究分布式鎖的問題,所以僅僅模擬了場景
// 這里使用自旋的方式,防止訪問信息丟失
sellTicket();
return "當前訪問人數過多,請稍后訪問...";
}
String ticketStr = stringRedisTemplate.opsForValue().get("ticket");
int ticket = 0;
if (null != ticketStr) {
ticket = Integer.parseInt(ticketStr);
}
if (ticket > 0) {
int ticketNew = ticket - 1;
stringRedisTemplate.opsForValue().set("ticket", String.valueOf(ticketNew));
logger.info("當前票的庫存為:" + ticketNew);
} else {
logger.info("手速不夠呀,票已經賣光了...");
}
stringRedisTemplate.delete(lock);
return "搶票成功...";
}

分析2

上述的代碼在程序正常運行下不會出現票超賣的問題,但是我們需要考慮:

  • 如果程序運行中系統出現了異常,導致無法刪除lock?,就會造成死鎖的問題。也許有人馬上就會想到,使用 try{} finally {} ,在finally中進行刪除鎖的操作。

但是,如果是分布式架構,第一個服務器接收到請求,加了鎖,此時第二個服務器也接收到請求,setnx 命令失敗,需要執行return操作,根據finally的特性,執行return之前,需要先執行finally里的代碼,于是,第二個服務器把鎖給刪除了,程序中鎖失效了,肯定會出現票超賣等一系列問題。

  • 如果程序在運行中直接徹底死了(比如,程序員閑著沒事兒,來了個 kill -9;或者斷電),就算加了finally,finally也不能執行,還是會出現死鎖問題

解決方法:

給鎖加一個標識符,只允許自己來操作鎖,其他訪問程序不能操作鎖

還要給鎖加一個過期時間,這樣就算程序死了,當時間過期后,還是能夠繼續執行

public String sellTicket(){
String lock="lock"; // 鎖的鍵
String lockId = UUID.randomUUID().toString(); // 鎖的值:唯一標識
try{
// 如果成功設置這個值,證明目前該方法并沒有被操作,可以進行賣票操作
// 添加一個過期時間,暫定為 30秒,這里的操作具有原子性,如果過期時間設置失敗,鍵也會設置失敗
Boolean tag = stringRedisTemplate.opsForValue().setIfAbsent(lock, lockId, 30, TimeUnit.SECONDS);
if (!tag) { // 如果設置失敗,證明當前方法正在被執行,不允許再次執行
// 實際開發環境應該使用隊列來完成訪問操作,這里主要探究分布式鎖的問題,所以僅僅模擬了場景
// 不設置回調的話,訪問信息會丟失
sellTicket();
return "當前訪問人數過多,請稍后訪問...";
}
String ticketStr = stringRedisTemplate.opsForValue().get("ticket");
int ticket = 0;
if (null != ticketStr) {
ticket = Integer.parseInt(ticketStr);
}
if (ticket > 0) {
int ticketNew = ticket - 1;
stringRedisTemplate.opsForValue().set("ticket", String.valueOf(ticketNew));
logger.info("當前票的庫存為:" + ticketNew);
} else {
logger.info("手速不夠呀,票已經賣光了...");
}
} finally {
// 如果redis中的值,和當前的值一致,才允許刪除鎖。
if (lockId.equals(stringRedisTemplate.opsForValue().get(lock))) {
stringRedisTemplate.delete(lock);
}
}
return "搶票成功...";
}

分析3

寫到這里已經可以解決大部分問題了,但是還需要考慮一個問題:

如果程序運行的極慢(硬件處理慢或者進行了GC),導致30秒已經到了,鎖已經失效了,程序還沒有運行完成,這時候,就會有另一個線程總想鉆個空子,導致票的超賣問題。

這里我們可以使用 sleep 模擬一下

  ......
if (ticket > 0) {
try {
// 為了測試方便,過期時間和線程暫停時間都改成了3秒
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
int ticketNew = ticket - 1;
stringRedisTemplate.opsForValue().set("ticket", String.valueOf(ticketNew));
......
  • 這樣運行就會出現極其嚴重的超賣問題

那么該如何設置這個過期時間呢?繼續加大?這顯然是不合適的,因為無論多么大,總有可能出現問題。

解決方法

我們可以使用守護線程,來保證這個時間永不過期

public String sellTicket(){
String lock="lock"; // 鎖的鍵
String lockId = UUID.randomUUID().toString(); // 鎖的值:唯一標識
MyThread myThread = null; // 鎖的守護線程
try{
// 如果成功設置這個值,證明目前該方法并沒有被操作,可以進行賣票操作
// 添加一個過期時間,暫定為 3 秒,這里的操作具有原子性,如果過期時間設置失敗,鍵也會設置失敗
Boolean tag = stringRedisTemplate.opsForValue().setIfAbsent(lock, lockId, 3, TimeUnit.SECONDS);
if (!tag) { // 如果設置失敗,證明當前方法正在被執行,不允許再次執行
// 實際開發環境應該使用隊列來完成訪問操作,這里主要探究分布式鎖的問題,所以僅僅模擬了場景
// 不設置回調的話,訪問信息會丟失
sellTicket();
return "當前訪問人數過多,請稍后訪問...";
}

// 開啟守護線程, 每隔三分之一的時間,給鎖續命
myThread = new MyThread(lock);
myThread.setDaemon(true);
myThread.start();

String ticketStr = stringRedisTemplate.opsForValue().get("ticket");
int ticket = 0;
if (null != ticketStr) {
ticket = Integer.parseInt(ticketStr);
}
if (ticket > 0) {
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
int ticketNew = ticket - 1;
stringRedisTemplate.opsForValue().set("ticket", String.valueOf(ticketNew));
logger.info("當前票的庫存為:" + ticketNew);
} else {
logger.info("手速不夠呀,票已經賣光了...");
}
} finally {
// 如果redis中的值,和當前的值一致,才允許刪除鎖。
if (lockId.equals(stringRedisTemplate.opsForValue().get(lock))) {
// 程序運行結束,需要關閉守護線程
myThread.stop();
stringRedisTemplate.delete(lock);
logger.info("釋放鎖成功...");
}
}
return "搶票成功...";
}

/** 使用后臺線程進行續命
* 守護線程
* 在主線程下 如果有一個守護線程 這個守護線程的生命周期 跟主線程是同生死的
*/
class MyThread extends Thread{
String lock;
MyThread (String lock) {
this.lock = lock;
}

@Override
public void run(){
while (true) {
try {
// 三分之一的時間
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 假設線程還活著,就要給鎖續命
logger.info("線程續命ing...");
stringRedisTemplate.expire(lock, 3, TimeUnit.SECONDS);
}
}
}

總結

到這里,我們已經基本實現了redis分布式鎖,并且可以在高并發場景下正常運行。

需要注意的是,實現分布式鎖的代碼肯定不是最佳的,重要的是了解分布式鎖的實現原理,以及發現問題并解決問題的思路。

責任編輯:武曉燕 來源: Java技術指北
相關推薦

2020-07-30 09:35:09

Redis分布式鎖數據庫

2024-02-19 00:00:00

Redis分布式

2021-11-01 12:25:56

Redis分布式

2019-06-19 15:40:06

分布式鎖RedisJava

2024-05-08 10:20:00

Redis分布式

2022-09-22 13:28:34

Redis分布式鎖

2022-09-29 08:28:57

SpringRedis分布式

2022-04-14 07:56:30

公平鎖Java線程

2024-07-15 08:25:07

2023-03-06 08:14:48

MySQLRedis場景

2022-05-18 10:38:51

Redis分布式鎖數據

2023-08-21 19:10:34

Redis分布式

2022-01-06 10:58:07

Redis數據分布式鎖

2019-02-26 09:51:52

分布式鎖RedisZookeeper

2023-09-21 22:22:51

開發分布式鎖

2022-12-18 20:07:55

Redis分布式

2018-10-12 09:42:00

分布式鎖 Java多線

2022-09-19 08:17:09

Redis分布式

2024-10-07 10:07:31

2020-11-16 12:55:41

Redis分布式鎖Zookeeper
點贊
收藏

51CTO技術棧公眾號

久久久久久久久久久久久久一区 | 国产精品一区二区中文字幕| 亚洲在线一区二区三区| 国产综合动作在线观看| 怡红院成永久免费人全部视频| 五月久久久综合一区二区小说| 精品999在线播放| 激情综合网俺也去| 羞羞的网站在线观看| 91香蕉视频黄| 亚洲一区二区三区久久 | 日韩午夜高潮| 色婷婷综合久久久久中文字幕1| 亚洲乱妇老熟女爽到高潮的片| 伊人色综合一区二区三区影院视频| 国产精品视频看| 国产在线精品日韩| 国产免费福利视频| 玖玖精品视频| 91国内精品久久| 欧美日韩色视频| 网红女主播少妇精品视频| 91精品午夜视频| 老司机午夜av| а√天堂中文在线资源8| 亚洲丝袜自拍清纯另类| 秋霞在线观看一区二区三区| 囯产精品久久久久久| 麻豆精品国产91久久久久久| 欧美在线视频一区| 久久久久久天堂| 91久久久精品国产| 国产一区二区三区直播精品电影| 亚洲精品激情视频| 国产精品一区二区三区av | 亚洲中文字幕无码不卡电影| 男人天堂亚洲| 成人免费小视频| 亚洲激情一区二区| 成人精品一区二区三区校园激情| 91丝袜美腿高跟国产极品老师 | 91国内精品久久| 免费在线视频观看| 综合激情网站| 久久福利视频导航| jizzjizz日本少妇| 青青草国产免费一区二区下载| 精品视频偷偷看在线观看| 欧美图片自拍偷拍| 欧美三级一区| 日韩视频免费观看高清完整版| xxww在线观看| 日韩毛片在线| 欧美日韩精品一区二区在线播放| wwwwxxxx日韩| 国产精品久久亚洲不卡| 欧美午夜精品电影| jizz大全欧美jizzcom| 久久av影院| 欧美日韩国产a| 日韩av自拍偷拍| 欧美一区一区| 欧美xxxxxxxxx| 少妇被狂c下部羞羞漫画| 欧美aaaaa级| 亚洲色图偷窥自拍| 日本污视频网站| 亚洲精品国产偷自在线观看| 九九综合九九综合| 日韩欧美三级在线观看| 亚洲中字黄色| 国产精品久久久久久搜索| 伊人网免费视频| 国产乱子伦一区二区三区国色天香| 91精品在线国产| 国产91免费在线观看| 91在线免费视频观看| 日本高清不卡一区二区三| 在线免费看av| 一个色在线综合| 乱妇乱女熟妇熟女网站| 粉嫩一区二区三区| 欧美久久久久久蜜桃| 日本天堂在线播放| 国产午夜一区| 九九视频直播综合网| 国产成人自拍视频在线| 美女视频一区二区三区| 97人人澡人人爽| 日韩欧美在线观看一区二区| 国产精品丝袜久久久久久app| 国产91av视频在线观看| 免费毛片在线看片免费丝瓜视频 | 九九久久久久99精品| 日本一区二区欧美| 美女网站在线免费欧美精品| 成人免费看片网站| 黄色片免费在线| 亚洲精品一卡二卡| 无码人妻丰满熟妇区五十路百度| 日本国产亚洲| 亚洲精品永久免费| 九九热最新地址| 久久xxxx精品视频| 亚洲影院色无极综合| 手机看片福利在线观看| 亚洲欧美综合在线精品| 黄色一级在线视频| 亚洲伦理一区二区| 亚洲精品网站在线播放gif| 麻豆精品国产免费| 久久久水蜜桃av免费网站| 亚洲free性xxxx护士hd| 国产免费av高清在线| 性感美女极品91精品| 91福利免费观看| 九九视频免费观看视频精品| 欧美黑人性生活视频| 亚洲一区在线观| 久久影院电视剧免费观看| 久久人妻无码一区二区| 国产一区高清| 亚洲欧美999| 日韩成年人视频| 国产米奇在线777精品观看| 日本午夜精品一区二区三区| 波多野结衣在线高清| 欧美妇女性影城| 国产午夜精品久久久久久久久| 99国产精品久久久久久久| 91麻豆蜜桃| 国产1区在线| 欧美三级韩国三级日本三斤| 丰腴饱满的极品熟妇| aa亚洲婷婷| 国产精品久久波多野结衣| 岛国成人毛片| 欧美精选在线播放| 国产福利在线导航| 另类综合日韩欧美亚洲| 欧洲精品国产| 吞精囗交69激情欧美| 亚洲精品美女在线观看播放| 国产在线观看99| 成人小视频在线| 精品人妻大屁股白浆无码| 日韩免费成人| 欧美丰满少妇xxxxx| 国产黄色一区二区| 亚洲狠狠丁香婷婷综合久久久| 亚洲精品在线视频播放| 一精品久久久| 97超级碰碰| 国产盗摄精品一区二区酒店| 亚洲精品在线观| 国产精品第56页| 99久久精品免费看| 久久精品国产精品亚洲色婷婷| 日韩精品免费一区二区夜夜嗨| 国产69久久精品成人| 欧美一区二区三区少妇| 在线精品视频一区二区| 国产视频三区四区| 另类欧美日韩国产在线| 国产香蕉一区二区三区| 91欧美极品| 91国产精品视频在线| 黑人与亚洲人色ⅹvideos| 欧美制服丝袜第一页| 亚洲视频重口味| 国产激情91久久精品导航| av在线观看地址| 神马电影久久| 国产精品女主播| av免费看在线| 亚洲精品久久久一区二区三区 | 日韩国产欧美精品一区二区三区| 狠狠人妻久久久久久综合| 中文字幕av免费专区久久| 999久久久精品视频| 亚洲国产一区二区精品专区| 日韩av不卡在线播放| 国产高清日韩| 91a在线视频| 午夜免费视频在线国产| 欧美大片国产精品| 亚洲黄网在线观看| 日韩美女视频一区| 第四色在线视频| 麻豆成人久久精品二区三区小说| 久久观看最新视频| 亚洲精品国产精品粉嫩| 91色p视频在线| 依依综合在线| 精品少妇v888av| 国产小视频在线观看| 日韩一级视频免费观看在线| 欧美性猛交bbbbb精品| 中文字幕一区二区日韩精品绯色| 第四色在线视频| 久久国产精品区| 国产午夜福利在线播放| 亚洲欧美综合久久久| 久久一区二区三区欧美亚洲| japansex久久高清精品| 日本精品久久中文字幕佐佐木| 超碰人人在线| 国产小视频91| 午夜av免费观看| 91精品国产综合久久香蕉的特点 | 精品一区二区免费在线观看| 欧美日韩性生活片| 欧美激情1区2区| 亚洲国产日韩综合一区| 麻豆成人入口| 亚洲自拍在线观看| 亚洲精品无播放器在线播放| 日本中文字幕不卡免费| 黄页网站在线观看免费| 色老头一区二区三区在线观看| 青青草在线视频免费观看| 精品电影一区二区| 国产精品国产三级国产普通话对白| 欧美性高潮在线| 色播视频在线播放| 一区二区三区久久久| 熟女少妇a性色生活片毛片| 欧美激情一区二区三区| 国产熟妇久久777777| 成人国产亚洲欧美成人综合网| 国内av一区二区| 麻豆91精品91久久久的内涵| 日本美女高潮视频| 久久成人免费| 岳毛多又紧做起爽| 亚洲综合电影一区二区三区| 国产96在线 | 亚洲| 精品96久久久久久中文字幕无| 中文字幕一区二区三区最新| 91综合在线| 在线观看欧美激情| 色综合色综合| 一本色道婷婷久久欧美| 日韩一区二区三区免费播放| 神马欧美一区二区| 欧美美女视频| 亚洲国产一区二区在线| 日韩国产一区二区三区| 永久久久久久| 888久久久| 欧美一级中文字幕| 亚洲小说区图片区| 丰满少妇久久久| 国产亚洲精品自拍| 免费裸体美女网站| 日本人妖一区二区| 性生活免费在线观看| 国产在线视频精品一区| 亚欧精品在线视频| 国产宾馆实践打屁股91| 催眠调教后宫乱淫校园| av电影在线观看一区| 中文字幕一区二区人妻在线不卡| 2019国产精品| 日本综合在线观看| 一区二区三区中文字幕在线观看| 国产第100页| 欧美午夜片欧美片在线观看| 久久久久久av无码免费看大片| 欧美三级视频在线播放| aa视频在线免费观看| 精品国产a毛片| 欧美3p视频在线观看| 伊人成人开心激情综合网| 麻豆网站在线| 欧美极品少妇xxxxⅹ免费视频 | 国产日韩在线亚洲字幕中文| 老司机亚洲精品一区二区| 国产精品一区二区欧美| 国产99久久精品一区二区300| 亚洲人成影视在线观看| 亚洲一本视频| 狠狠热免费视频| 国产福利91精品一区二区三区| 免费成人深夜夜行p站| 中文字幕av一区二区三区免费看 | 欧美精品videosex| 欧美一级在线播放| 婷婷成人av| 久久这里精品国产99丫e6| 日韩激情图片| 东北少妇不带套对白| 捆绑调教美女网站视频一区| 岛国精品一区二区三区| 国产精品私房写真福利视频| 日韩av电影网址| 7777精品久久久大香线蕉| 香蕉国产在线视频| 久久精品视频在线| 亚洲妇女成熟| 99伊人久久| 日韩精品首页| www.日本在线播放| 激情欧美一区二区| 永久免费成人代码| 亚洲地区一二三色| 91禁在线观看| 亚洲欧美日韩一区在线| 日韩伦理电影网站| 国产免费一区二区三区在线观看| 欧美交a欧美精品喷水| 欧美xxxx吸乳| 琪琪一区二区三区| 中文字幕狠狠干| 亚洲不卡av一区二区三区| 国产精品无码在线播放| 亚洲色图校园春色| av漫画网站在线观看| 51国偷自产一区二区三区| 日韩系列欧美系列| 任你操这里只有精品| 91麻豆精品在线观看| 国产亚洲精品久久777777| 制服丝袜亚洲网站| av网站在线免费观看| 国产成人精品综合久久久| 牛牛影视久久网| 少妇av一区二区三区无码| 国产成人av一区二区三区在线观看| 国产又粗又硬视频| 一本色道久久综合狠狠躁的推荐 | 特级西西444www高清大视频| 精品无人国产偷自产在线| 97人人在线视频| 日韩国产高清一区| 亚洲大尺度视频| 中文字幕在线看视频国产欧美在线看完整 | 国产精品日韩二区| 91精品国产91久久久久久密臀 | 岛国精品在线| 日韩免费中文专区| 久久亚洲图片| 中文字幕在线观看免费高清| 色av综合在线| 加勒比一区二区三区在线| 欧美亚洲视频在线看网址| 欧美激情99| 成人综合视频在线| 337p粉嫩大胆噜噜噜噜噜91av | 欧美日本精品一区二区三区| 91在线看黄| 成人免费视频网| 一区二区日韩欧美| 免费观看一区二区三区| 亚洲国产va精品久久久不卡综合| 六月婷婷综合网| 97人人做人人爱| 婷婷国产精品| 成人在线观看a| 国产精品剧情在线亚洲| 国产伦精品一区二区三区四区| 久久中文精品视频| 一区视频网站| a在线视频观看| 中文字幕国产一区| 国产乱码精品一区二区| 欧美韩国理论所午夜片917电影| 欧美理伦片在线播放| 日本新janpanese乱熟| 国产精品国产三级国产普通话三级| 国产乱码久久久| 久久久久久网址| 国产精品一在线观看| 男女污污视频网站| 亚洲高清在线精品| 国产小视频福利在线| 成人伊人精品色xxxx视频| 亚洲伦理精品| 国产三级在线观看完整版| 欧美一区二区三区色| 欧美大胆a人体大胆做受| 日韩高清dvd| 国产a视频精品免费观看| youjizz在线视频| 日韩在线视频国产| 蜜臀av一区| 亚洲18在线看污www麻豆| 亚洲成人免费在线| av免费观看一区二区| 国产精品12| 日韩电影在线一区二区三区| 欧美日韩在线视频免费播放| 亚洲毛片一区二区| 99亚洲男女激情在线观看| 99re在线视频免费观看| **欧美大码日韩| 日本啊v在线| 91精品天堂| 青娱乐精品视频| 国产精品第108页|