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

SpringBoot+Redis分布式鎖:模擬搶單

開發 后端 分布式 Redis
本篇內容主要講解的是redis分布式鎖,這個在各大廠面試幾乎都是必備的,下面結合模擬搶單的場景來使用她。

本篇內容主要講解的是redis分布式鎖,這個在各大廠面試幾乎都是必備的,下面結合模擬搶單的場景來使用她;本篇不涉及到的redis環境搭建,快速搭建個人測試環境,這里建議使用docker;本篇內容節點如下:

jedis的nx生成鎖

  •  如何刪除鎖
  •  模擬搶單動作(10w個人開搶)
  •  jedis的nx生成鎖

對于java中想操作redis,好的方式是使用jedis,首先pom中引入依賴: 

  1. <dependency>  
  2.     <groupId>redis.clients</groupId>  
  3.     <artifactId>jedis</artifactId>  
  4. </dependency> 

對于分布式鎖的生成通常需要注意如下幾個方面:

  •  創建鎖的策略:redis的普通key一般都允許覆蓋,A用戶set某個key后,B在set相同的key時同樣能成功,如果是鎖場景,那就無法知道到底是哪個用戶set成功的;這里jedis的setnx方式為我們解決了這個問題,簡單原理是:當A用戶先set成功了,那B用戶set的時候就返回失敗,滿足了某個時間點只允許一個用戶拿到鎖。
  •  鎖過期時間:某個搶購場景時候,如果沒有過期的概念,當A用戶生成了鎖,但是后面的流程被阻塞了一直無法釋放鎖,那其他用戶此時獲取鎖就會一直失敗,無法完成搶購的活動;當然正常情況一般都不會阻塞,A用戶流程會正常釋放鎖;過期時間只是為了更有保障。

下面來上段setnx操作的代碼: 

  1. public boolean setnx(String key, String val) {  
  2.         Jedis jedis = null 
  3.         try { 
  4.              jedis = jedisPool.getResource();  
  5.             if (jedis == null) {  
  6.                 return false;  
  7.             }  
  8.             return jedis.set(key, val, "NX", "PX", 1000 * 60).  
  9.                     equalsIgnoreCase("ok");  
  10.         } catch (Exception ex) {  
  11.         } finally {  
  12.             if (jedis != null) {  
  13.                 jedis.close();  
  14.             }  
  15.         }  
  16.         return false;  
  17.     } 

這里注意點在于jedis的set方法,其參數的說明如:

  •  NX:是否存在key,存在就不set成功
  •  PX:key過期時間單位設置為毫秒(EX:單位秒)

setnx如果失敗直接封裝返回false即可,下面我們通過一個get方式的api來調用下這個setnx方法: 

  1. @GetMapping("/setnx/{key}/{val}")  
  2. public boolean setnx(@PathVariable String key, @PathVariable String val) {  
  3.      return jedisCom.setnx(key, val);  

訪問如下測試url,正常來說第一次返回了true,第二次返回了false,由于第二次請求的時候redis的key已存在,所以無法set成功

由上圖能夠看到只有一次set成功,并key具有一個有效時間,此時已到達了分布式鎖的條件。

如何刪除鎖

上面是創建鎖,同樣的具有有效時間,但是我們不能完全依賴這個有效時間,場景如:有效時間設置1分鐘,本身用戶A獲取鎖后,沒遇到什么特殊情況正常生成了搶購訂單后,此時其他用戶應該能正常下單了才對,但是由于有個1分鐘后鎖才能自動釋放,那其他用戶在這1分鐘無法正常下單(因為鎖還是A用戶的),因此我們需要A用戶操作完后,主動去解鎖: 

  1. public int delnx(String key, String val) {  
  2.         Jedis jedis = null 
  3.         try {  
  4.             jedis = jedisPool.getResource();  
  5.             if (jedis == null) {  
  6.                 return 0;  
  7.             }  
  8.             //if redis.call('get','orderkey')=='1111' then return redis.call('del','orderkey') else return 0 end  
  9.             StringBuilder sbScript = new StringBuilder();  
  10.             sbScript.append("if redis.call('get','").append(key).append("')").append("=='").append(val).append("'").  
  11.                     append(" then ").  
  12.                     append("    return redis.call('del','").append(key).append("')").  
  13.                     append(" else ").  
  14.                     append("    return 0").  
  15.                     append(" end");  
  16.             return Integer.valueOf(jedis.eval(sbScript.toString()).toString());  
  17.         } catch (Exception ex) {  
  18.         } finally {  
  19.             if (jedis != null) {  
  20.                 jedis.close();  
  21.             }  
  22.         }  
  23.         return 0;  
  24.     } 

這里也使用了jedis方式,直接執行lua腳本:根據val判斷其是否存在,如果存在就del;

其實個人認為通過jedis的get方式獲取val后,然后再比較value是否是當前持有鎖的用戶,如果是那最后再刪除,效果其實相當;只不過直接通過eval執行腳本,這樣避免多一次操作了redis而已,縮短了原子操作的間隔。(如有不同見解請留言探討);同樣這里創建個get方式的api來測試: 

  1. @GetMapping("/delnx/{key}/{val}")  
  2. public int delnx(@PathVariable String key, @PathVariable String val) {  
  3.    return jedisCom.delnx(key, val);  

注意的是delnx時,需要傳遞創建鎖時的value,因為通過et的value與delnx的value來判斷是否是持有鎖的操作請求,只有value一樣才允許del;

模擬搶單動作(10w個人開搶)

有了上面對分布式鎖的粗略基礎,我們模擬下10w人搶單的場景,其實就是一個并發操作請求而已,由于環境有限,只能如此測試;如下初始化10w個用戶,并初始化庫存,商品等信息,如下代碼: 

  1. //總庫存  
  2.     private long nKuCuen = 0 
  3.     //商品key名字  
  4.     private String shangpingKey = "computer_key" 
  5.     //獲取鎖的超時時間 秒  
  6.     private int timeout = 30 * 1000;  
  7.     @GetMapping("/qiangdan")  
  8.     public List<String> qiangdan() {  
  9.         //搶到商品的用戶  
  10.         List<String> shopUsers = new ArrayList<>();  
  11.         //構造很多用戶  
  12.         List<String> users = new ArrayList<>();  
  13.         IntStream.range(0, 100000).parallel().forEach(b -> {  
  14.             users.add("神牛-" + b);  
  15.         });  
  16.         //初始化庫存  
  17.         nKuCuen = 10 
  18.         //模擬開搶  
  19.         users.parallelStream().forEach(b -> {  
  20.             String shopUser = qiang(b);  
  21.             if (!StringUtils.isEmpty(shopUser)) {  
  22.                 shopUsers.add(shopUser);  
  23.             }  
  24.         });  
  25.         return shopUsers;  
  26.     } 

有了上面10w個不同用戶,我們設定商品只有10個庫存,然后通過并行流的方式來模擬搶購,如下搶購的實現: 

  1. /**  
  2.      * 模擬搶單動作  
  3.      *  
  4.      * @param b  
  5.      * @return  
  6.      */  
  7.     private String qiang(String b) {  
  8.         //用戶開搶時間  
  9.         long startTime = System.currentTimeMillis();  
  10.         //未搶到的情況下,30秒內繼續獲取鎖  
  11.         while ((startTime + timeout) >= System.currentTimeMillis()) {  
  12.             //商品是否剩余  
  13.             if (nKuCuen <= 0) {  
  14.                 break;  
  15.             }  
  16.             if (jedisCom.setnx(shangpingKey, b)) {  
  17.                 //用戶b拿到鎖  
  18.                 logger.info("用戶{}拿到鎖...", b);  
  19.                 try {  
  20.                     //商品是否剩余  
  21.                     if (nKuCuen <= 0) {  
  22.                         break;  
  23.                     }  
  24.                     //模擬生成訂單耗時操作,方便查看:神牛-50 多次獲取鎖記錄  
  25.                     try {  
  26.                         TimeUnit.SECONDS.sleep(1);  
  27.                     } catch (InterruptedException e) {  
  28.                         e.printStackTrace();  
  29.                     }  
  30.                     //搶購成功,商品遞減,記錄用戶  
  31.                     nKuCuen -1 
  32.                     //搶單成功跳出  
  33.                     logger.info("用戶{}搶單成功跳出...所剩庫存:{}", b, nKuCuen);  
  34.                     return b + "搶單成功,所剩庫存:" + nKuCuen;  
  35.                 } finally {  
  36.                     logger.info("用戶{}釋放鎖...", b);  
  37.                     //釋放鎖  
  38.                     jedisCom.delnx(shangpingKey, b);  
  39.                 }  
  40.             } else {  
  41.                 //用戶b沒拿到鎖,在超時范圍內繼續請求鎖,不需要處理  
  42. //                if (b.equals("神牛-50") || b.equals("神牛-69")) {  
  43. //                    logger.info("用戶{}等待獲取鎖...", b);  
  44. //                }  
  45.             }  
  46.         }  
  47.         return "";  
  48.     } 

這里實現的邏輯是:

  •  parallelStream():并行流模擬多用戶搶購
  •  (startTime + timeout) >= System.currentTimeMillis():判斷未搶成功的用戶,timeout秒內繼續獲取鎖
  •  獲取鎖前和后都判斷庫存是否還足夠
  •  jedisCom.setnx(shangpingKey, b):用戶獲取搶購鎖
  •  獲取鎖后并下單成功,最后釋放鎖:jedisCom.delnx(shangpingKey, b)

再來看下記錄的日志結果:

最終返回搶購成功的用戶:

 

 

責任編輯:龐桂玉 來源: Java團長
相關推薦

2019-06-19 15:40:06

分布式鎖RedisJava

2023-01-13 07:39:07

2023-08-21 19:10:34

Redis分布式

2022-01-06 10:58:07

Redis數據分布式鎖

2019-02-26 09:51:52

分布式鎖RedisZookeeper

2022-09-19 08:17:09

Redis分布式

2024-10-07 10:07:31

2020-11-16 12:55:41

Redis分布式鎖Zookeeper

2021-06-16 07:56:21

Redis分布式

2024-04-01 05:10:00

Redis數據庫分布式鎖

2019-07-16 09:22:10

RedisZookeeper分布式鎖

2022-06-16 08:01:24

redis分布式鎖

2023-08-27 22:13:59

Redisson分布式緩存

2021-07-26 11:09:46

Redis分布式技術

2020-07-30 09:35:09

Redis分布式鎖數據庫

2021-10-26 19:37:15

RedisRedis應用篇

2020-07-15 16:50:57

Spring BootRedisJava

2022-07-22 06:55:20

Redis分布式鎖

2021-03-10 09:54:06

Redis分布式

2023-03-01 08:07:51

點贊
收藏

51CTO技術棧公眾號

亚洲性av网站| 日韩激情av在线| 欧美xingq一区二区| 亚洲成人动漫精品| 成人激情av| 4438国产精品一区二区| 久久99国产精品视频| 欧美日韩色综合| 日本a级片在线播放| 日本一区高清| 日本在线不卡视频| av在线播放一区二区三区| 这里只有精品视频在线观看| 大西瓜av在线| eeuss影院www在线播放| 国产成人av一区二区三区在线| 青青草成人在线| 国产精品爽爽爽| 五月天婷婷丁香网| 国产66精品| 欧美日韩免费观看一区三区| 波多野结衣乳巨码无在线| 日韩一区二区三区久久| 色www亚洲国产阿娇yao| 国产欧美自拍一区| 91麻豆精品国产91久久久久久久久| 国产免费一区二区三区视频| 性欧美videos高清hd4k| 国产精品人成在线观看免费| 久久国产精品-国产精品| av中文字幕播放| 美腿丝袜亚洲色图| 日本一区二区不卡| 亚洲第一在线播放| 国内综合精品午夜久久资源| 久久午夜a级毛片| 日本一道本视频| 美女毛片一区二区三区四区| 亚洲激情视频在线观看| 在线观看免费视频国产| 欧美高清hd| 欧美一级高清片| 91大神免费观看| 白嫩亚洲一区二区三区| 欧美一区二区三区四区在线观看| 国产精品一区二区小说| 日韩另类视频| 在线观看免费成人| 中文久久久久久| 97人人做人人爽香蕉精品| 欧美亚洲一区二区三区四区| 又色又爽又高潮免费视频国产| 韩国美女久久| 91精品办公室少妇高潮对白| 黄色免费网址大全| 国产第一精品| 欧美精品在线一区二区三区| 看看黄色一级片| 国产精品欧美一区二区三区不卡| 3atv在线一区二区三区| 日韩av福利在线观看| 99re6热只有精品免费观看| 日韩aaa久久蜜桃av| 亚洲午夜电影网| 成人性免费视频| 日本乱码一区二区三区不卡| 日韩欧美在线视频| 日本888xxxx| 99视频这里有精品| 精品国产伦一区二区三区观看方式 | 一区二区三区三区在线| 在线视频自拍| 亚洲三级久久久| 日本久久久网站| 欧美少妇精品| 欧美天堂亚洲电影院在线播放| 手机在线国产视频| 综合中文字幕| 亚洲久久久久久久久久| 少妇的滋味中文字幕bd| 国产丝袜精品丝袜| 亚洲国产婷婷| 欧美亚洲日本网站| 国产精品免费无遮挡无码永久视频| 石原莉奈一区二区三区在线观看| 国产日韩欧美一二三区| 午夜精品久久久久久久99老熟妇| 99精品一区二区三区| 日产精品一线二线三线芒果| 毛片av在线| 五月激情综合色| 在线看的黄色网址| 国产亚洲成av人片在线观黄桃| 国产一区二区三区免费视频| 草视频在线观看| 久久精品一区| 99在线影院| xxxxx日韩| 亚洲不卡一区二区三区| 亚欧激情乱码久久久久久久久| 亚洲国产高清在线观看| 亚洲视频欧美视频| 国产一级黄色av| 青草av.久久免费一区| 亚洲免费一在线| 一级免费黄色录像| 99精品免费| 91久久久在线| 国产对白叫床清晰在线播放| 亚洲国产一区二区视频| 香蕉视频999| 私拍精品福利视频在线一区| 久青草国产97香蕉在线视频| 亚洲另类在线观看| 成人午夜视频免费看| 亚洲午夜精品福利| 345成人影院| 精品国产免费人成电影在线观看四季 | 涩涩涩久久久成人精品| 精品亚洲男同gayvideo网站| 欧美国产在线看| 久久国产精品色| 清纯唯美一区二区三区| 蜜桃视频动漫在线播放| 日韩你懂的在线播放| 一区二区三区在线播放视频| 玖玖国产精品视频| 精品免费一区二区三区蜜桃| 色噜噜狠狠狠综合欧洲色8| 欧美色国产精品| 男人天堂av电影| 午夜一区不卡| 久久精品aaaaaa毛片| 俄罗斯一级**毛片在线播放| 日韩一区二区三区电影在线观看 | 欧美熟妇另类久久久久久多毛| 成人羞羞网站入口| 国产精品户外野外| 成年在线观看免费人视频| 日韩欧美精品网址| 搡老熟女老女人一区二区| 亚洲国产欧美国产综合一区| 国产高清一区视频| 97人人爽人人澡人人精品| 精品免费一区二区三区| 国产亚洲精品码| 丁香啪啪综合成人亚洲小说 | 亚洲精品videossex少妇| 久久香蕉精品视频| 成人一区二区三区在线观看| 成年人看的毛片| 精品综合久久88少妇激情| 久久久久久美女| 五月天激情开心网| 日韩欧美中文字幕在线播放| a级大片在线观看| 日韩av一二三| 伊人久久av导航| 日韩精品一区二区三区中文在线| 欧美激情喷水视频| 五月激情丁香婷婷| 亚洲成人原创| 国产乱子伦精品| 成人国产电影在线观看| 亚洲第一中文字幕在线观看| 日韩欧美中文字幕一区二区| 91香蕉视频黄| 少妇高清精品毛片在线视频| 精品国产一级毛片| 91香蕉视频导航| 国产黄色小视频在线| 欧美日韩成人综合在线一区二区| 国产精品一区二区久激情瑜伽| 久久久免费av| 视频污在线观看| 欧美性生交xxxxxdddd| 中文字幕av久久爽一区| 久久aⅴ国产欧美74aaa| 麻豆md0077饥渴少妇| jizz性欧美23| 日本sm极度另类视频| 一广人看www在线观看免费视频| 91精品国产免费| 日韩欧美激情视频| 日本一区二区高清| 在线观看你懂的视频| 亚洲一区欧美二区| 91麻豆蜜桃| 国产女同在线观看| 国产网站一区二区| 天天干天天曰天天操| 国产麻豆综合| 一级黄色免费在线观看| 日本妇女一区| 91精品国产综合久久久久久久久| 黄色大片在线| 中文字幕日韩欧美在线视频| 亚洲第一天堂影院| 欧美亚洲一区二区在线| 国产一级理论片| 中文字幕一区免费在线观看| 精品人妻一区二区免费视频| 激情欧美日韩一区二区| 日本三级免费观看| 国产一区日韩一区| 亚洲精品影院| 亚洲va久久久噜噜噜久久| 91中文字幕在线观看| 日韩在线观看不卡| 1769国产精品| 日韩精品无码一区二区三区久久久| 日韩综合小视频| 国产精彩视频一区二区| 日韩中字在线| 精品久久久久久国产91| 亚洲黄色小说视频| 大尺度一区二区| 日韩成人精品视频在线观看| 久久午夜电影| 2018日日夜夜| 欧美精品一区二区三区久久久竹菊| 香蕉久久夜色| 伊人成综合网伊人222| 国产精品久久久久久久久婷婷 | 色中色综合网| 茄子视频成人在线观看| 欧美电影在线观看完整版| 99热在线播放| 日韩在线观看中文字幕| 成人黄色激情网| 91九色综合| 国产日韩视频在线观看| 91av一区| 国产精品中文字幕在线| 成人在线免费av| 国产精品老女人视频| 日本不卡一二三| 人九九综合九九宗合| 91av亚洲| 国产mv免费观看入口亚洲| 乱一区二区三区在线播放| 国产高清在线观看| 亚洲精品国精品久久99热| 亚洲AV无码精品色毛片浪潮| 欧美一区日韩一区| 国产ts变态重口人妖hd| 日韩一区二区视频| 午夜精品久久久久久久99 | 天堂av8在线| 久久aⅴ国产欧美74aaa| 久久精品久久99| 国产成人福利片| 日本不卡视频一区| 91在线国产观看| 成人激情五月天| 中文字幕亚洲区| 男的操女的网站| 亚洲午夜激情av| 亚洲视频免费播放| 日本精品一级二级| 一级特黄aa大片| 欧美一区二区三区思思人| 亚洲国产综合一区| 亚洲精品在线视频| www.在线播放| 欧美成人精品在线播放| h片视频在线观看| 欧洲美女7788成人免费视频| 亚洲最新无码中文字幕久久| 国产精品老牛影院在线观看| 国产黄色片免费| 精品福利在线观看| 日批视频免费在线观看| 欧美曰成人黄网| 国产口爆吞精一区二区| 欧美mv日韩mv亚洲| 日韩精品视频无播放器在线看 | 日韩最新av在线| a视频在线观看| 韩国美女主播一区| 天然素人一区二区视频| 91视频免费在线观看| 婷婷综合电影| 在线无限看免费粉色视频| 在线看片一区| 日本在线观看免费视频| 懂色一区二区三区免费观看| 老熟妇一区二区| 亚洲精品免费在线播放| 中文字幕一区在线播放| 日韩一区二区三区视频在线| 欧美扣逼视频| 欧美情侣性视频| 91国内外精品自在线播放| 成人av资源网| 日韩激情图片| 免费看日本毛片| 韩国成人在线视频| 男生草女生视频| 亚洲高清视频在线| 国产一区二区三区黄片| 亚洲精品一区二区网址| 老司机av在线免费看| 清纯唯美日韩制服另类| 澳门精品久久国产| 自拍偷拍亚洲色图欧美| 久久成人亚洲| 香蕉视频污视频| 亚洲欧美日韩一区二区| 日批视频免费观看| 亚洲国内高清视频| 人妖欧美1区| 国产欧美日韩免费| 国产欧美日韩视频在线| 老太脱裤让老头玩ⅹxxxx| 国产一区亚洲一区| 免费看一级黄色| 欧洲一区二区三区在线| 三级毛片在线免费看| 午夜精品久久久久久久99热浪潮 | 91精品国产高潮对白| 9191久久久久久久久久久| 大片免费播放在线视频| 人人做人人澡人人爽欧美| 网曝91综合精品门事件在线| 国产欧美日韩网站| 国产成人精品免费网站| 中文字幕另类日韩欧美亚洲嫩草| 欧美老女人第四色| 在线观看国产原创自拍视频| 国产精品人人做人人爽| 狠狠综合久久av一区二区蜜桃 | 99亚洲伊人久久精品影院| 欧美日韩一区二区三区在线观看免| 亚洲伦伦在线| 香港三日本8a三级少妇三级99| 亚洲电影一级黄| 欧美特级特黄aaaaaa在线看| 久久久久久国产| 国产另类在线| 男人添女人下部高潮视频在观看| 不卡一区中文字幕| 日韩av片在线播放| 亚洲精品电影网| 在线人成日本视频| 日本不卡一区二区三区在线观看| 老司机精品久久| 少妇视频在线播放| 欧美丰满少妇xxxxx高潮对白| 欧美三级电影一区二区三区| 91免费版网站入口| 综合天堂av久久久久久久| 国偷自产av一区二区三区麻豆| 亚洲综合999| 凸凹人妻人人澡人人添| 欧美制服第一页| 不卡在线一区| 一二三级黄色片| 夜色激情一区二区| 无套内谢的新婚少妇国语播放| 日韩av片永久免费网站| 欧美色图国产精品| 九九久久久久久| 亚洲v精品v日韩v欧美v专区| 三级视频在线播放| 国产在线日韩在线| 久久久久久久久久久妇女| 日本wwwwwww| 色综合天天做天天爱| 午夜不卡视频| 国产精品二区三区四区| 久久久久免费| 永久免费看mv网站入口| 亚洲精品在线三区| 成人啊v在线| mm131午夜| 91免费国产视频网站| 亚洲天堂久久久久| 久久乐国产精品| 欧美自拍偷拍| 欧美做受高潮中文字幕| 在线免费精品视频| av色综合久久天堂av色综合在| 精品视频在线观看| 激情六月婷婷久久| 欧美在线观看不卡| 久久国产精品网站| 久久99青青| 国产亚洲色婷婷久久| 色哟哟国产精品免费观看| а√天堂在线官网| 日本亚洲导航| 国产成a人亚洲精| 正在播放亚洲精品| 海角国产乱辈乱精品视频| 色欧美自拍视频| 国产精品揄拍100视频| 日韩欧美aaaaaa| 九七电影院97理论片久久tvb| 国产极品在线视频|