婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > 詳解利用redis + lua解決搶紅包高并發的問題

詳解利用redis + lua解決搶紅包高并發的問題

熱門標簽:小紅書怎么地圖標注店 太原營銷外呼系統 最簡單的百度地圖標注 玄武湖地圖標注 竹間科技AI電銷機器人 地圖標注費用 西藏教育智能外呼系統價格 地圖標注如何即時生效 百度商家地圖標注怎么做

搶紅包的需求分析

搶紅包的場景有點像秒殺,但是要比秒殺簡單點。

因為秒殺通常要和庫存相關。而搶紅包則可以允許有些紅包沒有被搶到,因為發紅包的人不會有損失,沒搶完的錢再退回給發紅包的人即可。

另外像小米這樣的搶購也要比淘寶的要簡單,也是因為像小米這樣是一個公司的,如果有少量沒有搶到,則下次再搶,人工修復下數據是很簡單的事。而像淘寶這么多商品,要是每一個都存在著修復數據的風險,那如果出故障了則很麻煩。

基于redis的搶紅包方案

下面介紹一種基于Redis的搶紅包方案。

把原始的紅包稱為大紅包,拆分后的紅包稱為小紅包。

1.小紅包預先生成,插到數據庫里,紅包對應的用戶ID是null。生成算法見另一篇文章:https://www.jb51.net/article/98620.htm

2.每個大紅包對應兩個redis隊列,一個是未消費紅包隊列,另一個是已消費紅包隊列。開始時,把未搶的小紅包全放到未消費紅包隊列里。

未消費紅包隊列里是json字符串,如{userId:'789', money:'300'}。

3.在redis中用一個map來過濾已搶到紅包的用戶。

4.搶紅包時,先判斷用戶是否搶過紅包,如果沒有,則從未消費紅包隊列中取出一個小紅包,再push到另一個已消費隊列中,最后把用戶ID放入去重的map中。

5.用一個單線程批量把已消費隊列里的紅包取出來,再批量update紅包的用戶ID到數據庫里。

上面的流程是很清楚的,但是在第4步時,如果是用戶快速點了兩次,或者開了兩個瀏覽器來搶紅包,會不會有可能用戶搶到了兩個紅包?

為了解決這個問題,采用了lua腳本方式,讓第4步整個過程是原子性地執行。

下面是在redis上執行的Lua腳本:

-- 函數:嘗試獲得紅包,如果成功,則返回json字符串,如果不成功,則返回空 
-- 參數:紅包隊列名, 已消費的隊列名,去重的Map名,用戶ID 
-- 返回值:nil 或者 json字符串,包含用戶ID:userId,紅包ID:id,紅包金額:money 
 
-- 如果用戶已搶過紅包,則返回nil 
if rediscall('hexists', KEYS[3], KEYS[4]) ~= 0 then 
 return nil 
else 
 -- 先取出一個小紅包 
 local hongBao = rediscall('rpop', KEYS[1]); 
 if hongBao then 
  local x = cjsondecode(hongBao); 
  -- 加入用戶ID信息 
  x['userId'] = KEYS[4]; 
  local re = cjsonencode(x); 
  -- 把用戶ID放到去重的set里 
  rediscall('hset', KEYS[3], KEYS[4], KEYS[4]); 
  -- 把紅包放到已消費隊列里 
  rediscall('lpush', KEYS[2], re); 
  return re; 
 end 
end 
return nil 

下面是測試代碼:

public class TestEval { 
  static String host = "localhost"; 
  static int honBaoCount = 1_0_0000; 
   
  static int threadCount = 20; 
   
  static String hongBaoList = "hongBaoList"; 
  static String hongBaoConsumedList = "hongBaoConsumedList"; 
  static String hongBaoConsumedMap = "hongBaoConsumedMap"; 
   
  static Random random = new Random(); 
   
// -- 函數:嘗試獲得紅包,如果成功,則返回json字符串,如果不成功,則返回空 
// -- 參數:紅包隊列名, 已消費的隊列名,去重的Map名,用戶ID 
// -- 返回值:nil 或者 json字符串,包含用戶ID:userId,紅包ID:id,紅包金額:money 
  static String tryGetHongBaoScript =  
//     "local bConsumed = rediscall('hexists', KEYS[3], KEYS[4]);\n" 
//     + "print('bConsumed:' ,bConsumed);\n" 
      "if rediscall('hexists', KEYS[3], KEYS[4]) ~= 0 then\n" 
      + "return nil\n" 
      + "else\n" 
      + "local hongBao = rediscall('rpop', KEYS[1]);\n" 
//     + "print('hongBao:', hongBao);\n" 
      + "if hongBao then\n" 
      + "local x = cjsondecode(hongBao);\n" 
      + "x['userId'] = KEYS[4];\n" 
      + "local re = cjsonencode(x);\n" 
      + "rediscall('hset', KEYS[3], KEYS[4], KEYS[4]);\n" 
      + "rediscall('lpush', KEYS[2], re);\n" 
      + "return re;\n" 
      + "end\n" 
      + "end\n" 
      + "return nil"; 
  static StopWatch watch = new StopWatch(); 
   
  public static void main(String[] args) throws InterruptedException { 
//   testEval(); 
    generateTestData(); 
    testTryGetHongBao(); 
  } 
   
  static public void generateTestData() throws InterruptedException { 
    Jedis jedis = new Jedis(host); 
    jedisflushAll(); 
    final CountDownLatch latch = new CountDownLatch(threadCount); 
    for(int i = 0; i  threadCount; ++i) { 
      final int temp = i; 
      Thread thread = new Thread() { 
        public void run() { 
          Jedis jedis = new Jedis(host); 
          int per = honBaoCount/threadCount; 
          JSONObject object = new JSONObject(); 
          for(int j = temp * per; j  (temp+1) * per; j++) { 
            objectput("id", j); 
            objectput("money", j); 
            jedislpush(hongBaoList, objecttoJSONString()); 
          } 
          latchcountDown(); 
        } 
      }; 
      threadstart(); 
    } 
    latchawait(); 
  } 
   
  static public void testTryGetHongBao() throws InterruptedException { 
    final CountDownLatch latch = new CountDownLatch(threadCount); 
    Systemerrprintln("start:" + SystemcurrentTimeMillis()/1000); 
    watchstart(); 
    for(int i = 0; i  threadCount; ++i) { 
      final int temp = i; 
      Thread thread = new Thread() { 
        public void run() { 
          Jedis jedis = new Jedis(host); 
          String sha = jedisscriptLoad(tryGetHongBaoScript); 
          int j = honBaoCount/threadCount * temp; 
          while(true) { 
            Object object = jediseval(tryGetHongBaoScript, 4, hongBaoList, hongBaoConsumedList, hongBaoConsumedMap, "" + j); 
            j++; 
            if (object != null) { 
//             Systemoutprintln("get hongBao:" + object); 
            }else { 
              //已經取完了 
              if(jedisllen(hongBaoList) == 0) 
                break; 
            } 
          } 
          latchcountDown(); 
        } 
      }; 
      threadstart(); 
    } 
     
    latchawait(); 
    watchstop(); 
     
    Systemerrprintln("time:" + watchgetTotalTimeSeconds()); 
    Systemerrprintln("speed:" + honBaoCount/watchgetTotalTimeSeconds()); 
    Systemerrprintln("end:" + SystemcurrentTimeMillis()/1000); 
  } 
} 

測試結果20個線程,每秒可以搶2.5萬個,足以應付絕大部分的搶紅包場景。

如果是真的應付不了,拆分到幾個redis集群里,或者改為批量搶紅包,也足夠應付。

總結:

redis的搶紅包方案,雖然在極端情況下(即redis掛掉)會丟失一秒的數據,但是卻是一個擴展性很強,足以應付高并發的搶紅包方案。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • Nginx+Lua+Redis構建高并發Web應用
  • Redis實現高并發計數器
  • 如何利用Redis鎖解決高并發問題詳解
  • Redis瞬時高并發秒殺方案總結
  • PHP實現Redis單據鎖以及防止并發重復寫入
  • jedispool連redis高并發卡死的問題
  • 使用lua+redis解決發多張券的并發問題

標簽:澳門 唐山 林芝 揚州 香港 景德鎮 廣東 贛州

巨人網絡通訊聲明:本文標題《詳解利用redis + lua解決搶紅包高并發的問題》,本文關鍵詞  詳解,利用,redis,lua,解決,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《詳解利用redis + lua解決搶紅包高并發的問題》相關的同類信息!
  • 本頁收集關于詳解利用redis + lua解決搶紅包高并發的問題的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    亚洲精品在线免费观看视频| 亚洲一区二区视频在线| 亚洲最大成人综合| 国产精品123| 欧美日韩亚州综合| 中文字幕亚洲精品在线观看| 狠狠狠色丁香婷婷综合久久五月| 日本精品免费观看高清观看| 国产亚洲精品资源在线26u| 婷婷成人综合网| 欧美综合久久久| 中文字幕av一区二区三区| 精东粉嫩av免费一区二区三区| 欧美性受xxxx| 亚洲人成精品久久久久久| 波多野结衣在线一区| 国产夜色精品一区二区av| 奇米四色…亚洲| 欧美一区二区免费视频| 亚洲成人自拍网| 欧美日本在线视频| 亚洲国产精品久久久久秋霞影院| 91亚洲精华国产精华精华液| 国产精品女主播在线观看| 福利一区二区在线观看| 国产午夜精品理论片a级大结局 | 亚洲日本一区二区三区| 成人av先锋影音| 亚洲欧洲精品一区二区三区 | 麻豆国产精品777777在线| 欧美日韩精品一区视频| 亚洲成人av电影| 欧美日韩国产美女| 日韩高清不卡在线| 日韩欧美国产一区二区三区 | 欧美日韩精品一区视频| 天天亚洲美女在线视频| 欧美精品丝袜中出| 免费看黄色91| 久久久精品国产99久久精品芒果 | 成人激情图片网| 国产精品久久久一本精品| 成人黄色一级视频| 一区二区在线看| 欧美顶级少妇做爰| 韩国成人精品a∨在线观看| 国产亚洲精品免费| 99久久伊人久久99| 亚洲国产精品影院| 欧美成人猛片aaaaaaa| 国产二区国产一区在线观看| 国产精品初高中害羞小美女文| 91蝌蚪porny九色| 天天综合天天做天天综合| 2020国产精品自拍| 97se亚洲国产综合自在线不卡| 亚洲高清在线视频| 久久久久久久久久久久久久久99 | 不卡的av电影在线观看| 亚洲精品综合在线| 日韩欧美国产三级| 粉嫩绯色av一区二区在线观看 | 亚洲欧洲制服丝袜| 日韩亚洲欧美一区| av高清久久久| 日韩成人av影视| 欧美一区二区久久久| 亚洲午夜久久久| 久久久美女毛片| 欧美日韩不卡视频| 播五月开心婷婷综合| 久久精品国产99国产| 一区二区三区国产精品| 久久综合色天天久久综合图片| 97超碰欧美中文字幕| 麻豆极品一区二区三区| 亚洲国产色一区| 国产精品久久久久久亚洲毛片 | 综合激情成人伊人| 精品国产亚洲在线| 欧美人狂配大交3d怪物一区| 成人h动漫精品一区二区| 久久精品国产精品亚洲精品| 亚洲成人一区二区在线观看| 国产精品久久久爽爽爽麻豆色哟哟 | 日本美女一区二区三区视频| 亚洲精品国产a| 国产清纯美女被跳蛋高潮一区二区久久w| 在线欧美一区二区| 91亚洲精品乱码久久久久久蜜桃| 国内欧美视频一区二区| 日本不卡一区二区三区高清视频| 一区二区三区 在线观看视频 | 国产午夜精品一区二区三区嫩草| 欧美精品第1页| 欧日韩精品视频| av电影天堂一区二区在线观看| 国产一区二区三区四区五区入口| 三级亚洲高清视频| 午夜精品久久久久影视| 一区二区三区中文在线| 亚洲男同性视频| 国产精品白丝在线| 国产精品传媒入口麻豆| 中文字幕中文在线不卡住| 国产亚洲欧洲997久久综合| 精品国产乱码久久久久久蜜臀| 这里是久久伊人| 91精品中文字幕一区二区三区| 成人动漫av在线| 成人黄色网址在线观看| 成人激情开心网| 99国产精品一区| 91麻豆免费观看| 欧洲国产伦久久久久久久| 一本一道综合狠狠老| 91久久精品日日躁夜夜躁欧美| 97精品国产97久久久久久久久久久久| 9色porny自拍视频一区二区| 91麻豆国产香蕉久久精品| 在线观看日韩国产| 欧美猛男gaygay网站| 日韩免费观看高清完整版在线观看| 日韩欧美亚洲另类制服综合在线| 日韩视频一区二区在线观看| 欧美变态凌虐bdsm| 国产欧美日韩久久| 亚洲欧美成人一区二区三区| 亚洲品质自拍视频| 亚洲1区2区3区4区| 国产美女娇喘av呻吟久久| av电影一区二区| 欧美视频三区在线播放| 日韩三级视频在线看| 久久久综合视频| 亚洲欧洲成人av每日更新| 午夜电影久久久| 久久aⅴ国产欧美74aaa| 99精品视频一区二区| 91麻豆精品国产91久久久久久| 久久精品网站免费观看| 亚洲女人小视频在线观看| 男人操女人的视频在线观看欧美| 国v精品久久久网| 欧美日韩一区二区在线观看视频| 欧美成人女星排行榜| 亚洲色图视频网站| 裸体在线国模精品偷拍| 91尤物视频在线观看| 精品日韩在线观看| 一区二区高清视频在线观看| 国内成人精品2018免费看| 色综合久久天天| 亚洲精品一区二区三区影院| 亚洲欧美一区二区视频| 久久精品噜噜噜成人88aⅴ| 91免费在线视频观看| 26uuu久久综合| 亚洲在线成人精品| 成人av电影免费观看| 日韩亚洲欧美成人一区| 一级做a爱片久久| 丰满放荡岳乱妇91ww| 日韩一级黄色大片| 亚洲妇女屁股眼交7| av影院午夜一区| www国产成人| 久久精品99国产精品日本| 欧美精品色综合| 亚洲一区二区精品视频| 99精品欧美一区二区三区综合在线| 欧美成人一区二区| 日本欧美一区二区在线观看| 91精品福利视频| 亚洲乱码国产乱码精品精可以看| 国产一区二区三区黄视频 | 美女视频网站黄色亚洲| 日本高清无吗v一区| 亚洲欧洲精品一区二区三区不卡| 国产成人精品在线看| 久久色.com| 国内精品在线播放| 26uuu国产日韩综合| 九九久久精品视频| 欧美不卡一区二区三区| 久久黄色级2电影| 日韩一区二区免费电影| 日本免费在线视频不卡一不卡二| 欧美日韩一本到| 日本三级韩国三级欧美三级| 欧美日韩国产综合一区二区三区| 免费精品视频在线| 久久综合九色综合97_久久久| 欧美日韩一区二区三区四区| 99精品在线观看视频| 激情丁香综合五月| 国产大陆精品国产| 久久久久久久精| 国产成人免费视频精品含羞草妖精| 精品国产一区二区三区av性色|