精品欧美一区二区三区在线观看 _久久久久国色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技術棧公眾號

亚洲精品日日夜夜| 国产精品vip| 欧美日韩在线精品一区二区三区激情 | 成人免费视频视频在| 国产主播在线播放| 精品国产网站| 精品久久人人做人人爰| 国产男女激情视频| 国产三级在线播放| 99免费精品视频| 国产原创欧美精品| 日韩 欧美 综合| 久久人人88| 亚洲精品午夜精品| 亚洲国产日韩在线一区| 欧美亚洲韩国| 亚洲一区二区三区四区在线观看| 欧美国产综合视频| 亚洲春色一区二区三区| 久久综合五月| 国内精品一区二区三区| 波兰性xxxxx极品hd| 亚洲精品白浆高清| 欧美成人video| 色一情一区二区| 伊人久久国产| 亚洲成人在线网站| 五月天av影院| jzzjzzjzz亚洲成熟少妇| caoporen国产精品视频| 91pron在线| 国产又黄又大又爽| 日韩av一二三| 2019中文字幕在线免费观看| 九九热精品在线观看| 999国产精品永久免费视频app| 亚洲性线免费观看视频成熟| 妖精视频一区二区| 午夜视频一区二区在线观看| 欧美日本在线视频| 日韩av片网站| 成人看片在线观看| 色综合天天综合狠狠| 免费拍拍拍网站| 日本色护士高潮视频在线观看| 国产精品妹子av| 亚洲国产一区在线| 国产特黄在线| 久久亚洲精品国产精品紫薇| 91色精品视频在线| 国产精品热久久| 久久国产综合精品| 国产欧美日韩丝袜精品一区| 国产成人av免费| 日韩电影网1区2区| 国产精品国产三级国产aⅴ9色| 五月天婷婷激情| 蜜桃久久av| 国产成人精品电影| 国产情侣呻吟对白高潮| 日本成人在线不卡视频| 国产精品免费在线免费| 中文字幕在线观看国产| 久久综合综合久久综合| 成人在线免费观看视视频| 国产口爆吞精一区二区| 国产一区二区三区免费| 91久久国产综合久久蜜月精品| 国产免费不卡视频| 国产精品18久久久| 国产精品视频一区二区三区经| 亚洲国产精品成人久久蜜臀| 不卡电影一区二区三区| 久久国产精品久久| 成人资源www网在线最新版| 欧美高清在线精品一区| www.亚洲一区二区| 免费在线观看av电影| 亚洲不卡一区二区三区| 日韩精品视频一区二区在线观看| 三级成人黄色影院| 欧美精品日韩一区| 精品国产aⅴ一区二区三区东京热| 成人h动漫精品一区二区器材| 亚洲国产精品va在线看黑人动漫| 37p粉嫩大胆色噜噜噜| 欧美熟乱15p| 美女撒尿一区二区三区| 黄色片视频网站| 蜜臀av一级做a爰片久久| 91手机视频在线观看| 欧美熟妇乱码在线一区| 国产欧美综合色| 中文字幕精品在线播放| 偷拍自拍在线看| 欧美精品三级在线观看| 97人妻精品一区二区三区免| 欧美高清在线| 91精品国产91久久久久久久久 | 91福利国产在线观看菠萝蜜| 偷窥国产亚洲免费视频| 色一情一区二区| 女同另类激情重口| 色爱av美腿丝袜综合粉嫩av| 精品少妇久久久| 视频一区二区三区入口| 成人h视频在线观看| 国产毛片av在线| 亚洲综合在线视频| 男女污污的视频| 国产精品毛片久久久| 一区二区欧美久久| 国产精彩视频在线| 黄网站免费久久| 欧美极品一区| h片精品在线观看| 欧美日韩久久一区| 女尊高h男高潮呻吟| 欧美在线黄色| 国产久一一精品| 日韩美女一级视频| 夜色激情一区二区| 岛国av免费在线| 精品久久视频| 91高潮在线观看| 亚洲成熟女性毛茸茸| 中文字幕一区二| 中文字幕第80页| 香蕉久久夜色精品国产使用方法 | 日韩精品视频在线免费观看| www.超碰在线观看| 久草中文综合在线| 亚洲国产欧洲综合997久久 | 91日韩久久| 国产欧美久久久久久久久| 91久久线看在观草草青青| 亚洲中文字幕无码一区| 欧美va天堂在线| 91精品综合久久久久久五月天| 国产黄色片在线观看| 日韩欧美在线中文字幕| 亚洲国产第一区| 亚洲国产高清视频| 国产一区二区免费在线观看| 欧美xxx黑人xxx水蜜桃| 日韩欧美中文字幕公布| 色哟哟一一国产精品| 久久www免费人成看片高清| 色999五月色| 成人免费网站www网站高清| 亚洲欧美日韩国产中文| 日本中文字幕在线| 久久久91精品国产一区二区精品 | 成人性生活视频| 日韩精品免费在线观看| 欧美精品一二三四区| 久久丝袜美腿综合| mm1313亚洲国产精品无码试看| 国产一区二区精品福利地址| 国产精品高潮呻吟视频| 免费高清在线观看| 欧美一区二区三区精品| 青娱乐国产在线| 成a人片国产精品| 国产精品秘入口18禁麻豆免会员 | 婷婷视频在线播放| 综合视频一区| 日本不卡视频在线播放| 成人精品一区二区三区免费| 91麻豆精品国产91久久久使用方法| 亚洲xxxx3d动漫| 成人av在线一区二区| 欧美成人xxxxx| 色综合久久一区二区三区| 成人免费淫片视频软件| 免费不卡av| 亚洲欧洲黄色网| 国产精品熟女久久久久久| 一区二区三区在线高清| 水蜜桃av无码| 日韩精品亚洲专区| www国产无套内射com| 日韩激情啪啪| 成人妇女免费播放久久久| 蜜臀av国内免费精品久久久夜夜| 日韩成人中文字幕在线观看| 少妇无套内谢久久久久| 一区二区三区四区乱视频| 欧美无人区码suv| 麻豆精品在线播放| a天堂资源在线观看| 国产欧美一区二区三区精品观看| 91久久精品国产| 蜜桃麻豆av在线| 中文字幕亚洲欧美日韩2019| 东京干手机福利视频| 色国产综合视频| 三级影片在线看| 91免费观看国产| www一区二区www免费| 成人国产免费电影| 丁香天五香天堂综合| 国产精品免费看久久久无码| 免费av一区二区三区四区| 国产精品第一区| 无遮挡的视频在线观看 | 亚洲网一区二区三区| 2019亚洲日韩新视频| a黄色在线观看| 亚洲国产精彩中文乱码av| 丰满人妻一区二区三区四区| 亚洲丝袜另类动漫二区| 337p日本欧洲亚洲大胆张筱雨| 久久久成人网| 黄色三级中文字幕| 成人精品久久| 成人影片在线播放| 欧美成人福利| 97在线免费观看| jyzzz在线观看视频| 欧美成人激情免费网| 91色在线播放| 色哟哟国产精品| 欧美成人精品激情在线视频| 国产精品久久777777| 亚洲精品乱码久久久久久久| 激情图片小说一区| 一本色道无码道dvd在线观看| 99国产精品视频免费观看一公开 | 国产精品成av人在线视午夜片 | 九九九久久久久久| www 日韩| 亚洲欧美日韩在线高清直播| 天堂网av在线播放| 欧美一区二区三区视频在线观看| 久久久黄色大片| 亚洲午夜精品17c| 希岛爱理中文字幕| 国产精品乱码人人做人人爱 | 国产偷国产偷精品高清尤物| 中文字幕乱视频| 美日韩一区二区| 免费男同深夜夜行网站| 美女精品一区| 欧美 日韩 国产一区| 国内视频精品| 黄网站色视频免费观看| 欧美黄在线观看| 国产美女视频免费| 久久精品国产亚洲夜色av网站| 午夜一区二区三区| 国产尤物久久久| 久久久神马电影| 成人知道污网站| 精品中文字幕一区| 国产精品午夜av| 国产精品二区三区| 欧美高清一级片| 国产精品日韩欧美一区二区三区 | 波多野结衣三级在线| av大片免费观看| 最新中文字幕一区二区三区| 精品无码在线观看| 国产人久久人人人人爽| 中文精品在线观看| 国产欧美一区二区精品久导航| 亚洲熟妇无码av| 久久久久综合网| 精品国产av色一区二区深夜久久| 99精品国产91久久久久久| 亚洲调教欧美在线| 91麻豆产精品久久久久久| 久久久久久久久久久久| 国产亚洲欧洲997久久综合| 精品人妻无码一区二区三区换脸| 久久久久久亚洲综合影院红桃| 免费看日本黄色片| 国产精品乱码人人做人人爱 | 中文字幕第一区| 国产大屁股喷水视频在线观看| 国产精品国产a级| 男女做暖暖视频| 一区二区在线观看不卡| 精品在线视频免费| 欧美在线视频不卡| 911美女片黄在线观看游戏| 3d动漫精品啪啪一区二区竹菊| www.黄色小说.com| 亚洲国产精品免费| 久久精品蜜桃| 日韩视频免费观看| 蜜桃视频在线网站| 日韩av第一页| 福利一区二区| 成人精品一区二区三区电影免费 | 欧美一区二区不卡视频| 男人天堂综合网| 一本色道久久88综合日韩精品| 视频在线观看入口黄最新永久免费国产| 色综合导航网站| 在线男人天堂| 国产主播欧美精品| 国产成人aa在线观看网站站| 国产日韩一区欧美| 精品国产乱码久久久| 法国空姐在线观看免费| 国产欧美日韩一级| 免费黄频在线观看| av中文字幕亚洲| 五月天av网站| 色综合中文综合网| 国产三级第一页| 伊人久久精品视频| 7777kkk亚洲综合欧美网站| 国产精品xxx视频| 天堂一区二区三区四区| 自拍视频一区二区三区| 国产一级久久| 色噜噜狠狠一区二区| 久久久久久久综合日本| 私库av在线播放| 在线观看国产日韩| 头脑特工队2免费完整版在线观看| 中文字幕日韩av| 免费看男女www网站入口在线 | 国产又粗又猛又黄又爽| 日韩精品免费一线在线观看| 亚洲电影视频在线| 国产精品久久久久久久天堂| 狠狠一区二区三区| 一级全黄肉体裸体全过程| 一本色道久久综合亚洲精品不卡 | 色综合久久久久无码专区| 日韩av一二三| 动漫美女无遮挡免费| 国产亚洲精品资源在线26u| 992tv在线成人免费观看| 人妻激情另类乱人伦人妻 | 538在线视频| 无吗不卡中文字幕| 国产精品亚洲自拍| 中文字幕在线观看播放| 欧美激情精品久久久久久黑人 | 亚洲午夜小视频| 国产美女久久精品香蕉69| 人人鲁人人莫人人爱精品| 日韩美女主播视频| 人妻一区二区视频| www.超碰在线| 成人深夜视频在线观看| 久久露脸国产精品| 99精品一区二区三区无码吞精 | 制服丝袜亚洲网站| 五月激情婷婷综合| 另类人妖一区二区av| 黄色特一级视频| 麻豆国产精品视频| jizzjizzjizz国产| 精品婷婷伊人一区三区三| 激情在线视频| 国产精品视频地址| 色呦哟—国产精品| 色啦啦av综合| 亚洲三级网页| 日本高清一区| 国产一区二区精品| 中文字幕国产专区| 欧美亚洲日本国产| 岛国视频免费在线观看| 国产999精品| japanese国产在线观看| 制服丝袜中文字幕一区| 免费在线视频欧美| 91久久精品美女高潮| 中国成人一区| 91精品啪在线观看国产| 亚洲国产精品人人做人人爽| 日韩一级片免费| 57pao国产成人免费| 欧美偷窥清纯综合图区| 久久久久久久激情| 中文字幕不卡在线播放| www.色播.com| 91国产视频在线| 久久99精品久久久久久园产越南| 在线观看国产一级片| 亚洲另类色综合网站| 国产精品主播一区二区| 97香蕉超级碰碰久久免费的优势 | 亚洲最大成人网色| 一区二区亚洲| 久久精品国产亚洲AV成人婷婷| 5月丁香婷婷综合| 成人爽a毛片免费啪啪动漫| 欧洲一区二区日韩在线视频观看免费| 日本视频在线一区| 欧美一区二区三区爽爽爽| 亚洲视频欧美视频| 国产精品日本一区二区三区在线| 国产日韩欧美大片|