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

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

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

熱門標(biāo)簽:小紅書怎么地圖標(biāo)注店 太原營銷外呼系統(tǒng) 最簡單的百度地圖標(biāo)注 玄武湖地圖標(biāo)注 竹間科技AI電銷機(jī)器人 地圖標(biāo)注費(fèi)用 西藏教育智能外呼系統(tǒng)價(jià)格 地圖標(biāo)注如何即時(shí)生效 百度商家地圖標(biāo)注怎么做

搶紅包的需求分析

搶紅包的場景有點(diǎn)像秒殺,但是要比秒殺簡單點(diǎn)。

因?yàn)槊霘⑼ǔR蛶齑嫦嚓P(guān)。而搶紅包則可以允許有些紅包沒有被搶到,因?yàn)榘l(fā)紅包的人不會有損失,沒搶完的錢再退回給發(fā)紅包的人即可。

另外像小米這樣的搶購也要比淘寶的要簡單,也是因?yàn)橄裥∶走@樣是一個(gè)公司的,如果有少量沒有搶到,則下次再搶,人工修復(fù)下數(shù)據(jù)是很簡單的事。而像淘寶這么多商品,要是每一個(gè)都存在著修復(fù)數(shù)據(jù)的風(fēng)險(xiǎn),那如果出故障了則很麻煩。

基于redis的搶紅包方案

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

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

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

2.每個(gè)大紅包對應(yīng)兩個(gè)redis隊(duì)列,一個(gè)是未消費(fèi)紅包隊(duì)列,另一個(gè)是已消費(fèi)紅包隊(duì)列。開始時(shí),把未搶的小紅包全放到未消費(fèi)紅包隊(duì)列里。

未消費(fèi)紅包隊(duì)列里是json字符串,如{userId:'789', money:'300'}。

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

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

5.用一個(gè)單線程批量把已消費(fèi)隊(duì)列里的紅包取出來,再批量update紅包的用戶ID到數(shù)據(jù)庫里。

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

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

下面是在redis上執(zhí)行的Lua腳本:

-- 函數(shù):嘗試獲得紅包,如果成功,則返回json字符串,如果不成功,則返回空 
-- 參數(shù):紅包隊(duì)列名, 已消費(fèi)的隊(duì)列名,去重的Map名,用戶ID 
-- 返回值:nil 或者 json字符串,包含用戶ID:userId,紅包ID:id,紅包金額:money 
 
-- 如果用戶已搶過紅包,則返回nil 
if rediscall('hexists', KEYS[3], KEYS[4]) ~= 0 then 
 return nil 
else 
 -- 先取出一個(gè)小紅包 
 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]); 
  -- 把紅包放到已消費(fèi)隊(duì)列里 
  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(); 
   
// -- 函數(shù):嘗試獲得紅包,如果成功,則返回json字符串,如果不成功,則返回空 
// -- 參數(shù):紅包隊(duì)列名, 已消費(fèi)的隊(duì)列名,去重的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 { 
              //已經(jīng)取完了 
              if(jedisllen(hongBaoList) == 0) 
                break; 
            } 
          } 
          latchcountDown(); 
        } 
      }; 
      threadstart(); 
    } 
     
    latchawait(); 
    watchstop(); 
     
    Systemerrprintln("time:" + watchgetTotalTimeSeconds()); 
    Systemerrprintln("speed:" + honBaoCount/watchgetTotalTimeSeconds()); 
    Systemerrprintln("end:" + SystemcurrentTimeMillis()/1000); 
  } 
} 

測試結(jié)果20個(gè)線程,每秒可以搶2.5萬個(gè),足以應(yīng)付絕大部分的搶紅包場景。

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

總結(jié):

redis的搶紅包方案,雖然在極端情況下(即redis掛掉)會丟失一秒的數(shù)據(jù),但是卻是一個(gè)擴(kuò)展性很強(qiáng),足以應(yīng)付高并發(fā)的搶紅包方案。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • Nginx+Lua+Redis構(gòu)建高并發(fā)Web應(yīng)用
  • Redis實(shí)現(xiàn)高并發(fā)計(jì)數(shù)器
  • 如何利用Redis鎖解決高并發(fā)問題詳解
  • Redis瞬時(shí)高并發(fā)秒殺方案總結(jié)
  • PHP實(shí)現(xiàn)Redis單據(jù)鎖以及防止并發(fā)重復(fù)寫入
  • jedispool連redis高并發(fā)卡死的問題
  • 使用lua+redis解決發(fā)多張券的并發(fā)問題

標(biāo)簽:澳門 唐山 林芝 揚(yáng)州 香港 景德鎮(zhèn) 廣東 贛州

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《詳解利用redis + lua解決搶紅包高并發(fā)的問題》,本文關(guān)鍵詞  詳解,利用,redis,lua,解決,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《詳解利用redis + lua解決搶紅包高并發(fā)的問題》相關(guān)的同類信息!
  • 本頁收集關(guān)于詳解利用redis + lua解決搶紅包高并發(fā)的問題的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    99久久伊人精品| 中文字幕欧美一| 久久理论电影网| 国产99精品国产| 亚洲色图视频网| 欧美人与禽zozo性伦| 91片在线免费观看| 亚洲黄色免费电影| 视频一区视频二区中文| 欧美性猛交xxxx乱大交退制版| 国产精品天天看| 国产精品456| 中文字幕在线观看不卡视频| 欧美激情中文字幕一区二区| 成人动漫视频在线| 一区二区三区精品视频| 欧美日韩精品二区第二页| 一区二区三区免费在线观看| 欧美在线观看18| 国产精品久久久久久久久晋中| 国产一区不卡视频| 亚洲一区二区视频在线| 日韩视频一区二区三区在线播放| 美女国产一区二区| 国产日韩视频一区二区三区| 欧美私模裸体表演在线观看| 亚洲一区精品在线| 欧美日韩中字一区| 福利视频网站一区二区三区| 成人国产精品免费观看视频| 欧美亚一区二区| 偷拍与自拍一区| 日韩欧美一区二区免费| 色婷婷综合激情| 亚洲国产精品一区二区www| 欧美xxxx在线观看| 99精品视频一区二区三区| 亚洲地区一二三色| 欧美一区二区三区在线观看视频| 在线播放亚洲一区| 国产午夜精品一区二区| 91国偷自产一区二区三区观看| 精彩视频一区二区| 午夜精品123| 亚洲色图第一区| 欧美一级免费大片| 91免费观看视频在线| 精品一区二区综合| 综合激情网...| 在线一区二区视频| 国产成人午夜片在线观看高清观看| 亚洲欧美一区二区三区国产精品| 久久久国产综合精品女国产盗摄| 精品视频一区三区九区| 成人av网站免费观看| 国产精品大尺度| 精品剧情在线观看| 日韩免费观看高清完整版| eeuss鲁片一区二区三区在线观看| 91视频免费播放| 欧美无砖砖区免费| 99国产欧美另类久久久精品 | 亚洲精品在线免费观看视频| 日本特黄久久久高潮| 久久久99精品久久| 在线观看免费一区| 美国三级日本三级久久99| 久久综合精品国产一区二区三区| 精品理论电影在线观看| 777午夜精品视频在线播放| 波多野结衣的一区二区三区| 精品一区二区三区免费| 韩日精品视频一区| 狠狠色丁香婷婷综合| 欧美午夜不卡视频| 91视视频在线观看入口直接观看www| 亚洲欧美日韩人成在线播放| 日韩欧美的一区| 欧美精品欧美精品系列| av中文一区二区三区| 午夜欧美在线一二页| 中文字幕av一区二区三区| 91精品国产色综合久久久蜜香臀| 99re这里只有精品视频首页| 久久国产尿小便嘘嘘| 中文字幕在线一区免费| 日韩一级高清毛片| 亚洲制服欧美中文字幕中文字幕| 精品99一区二区| 欧美午夜精品久久久久久孕妇| 琪琪一区二区三区| 极品瑜伽女神91| 99久久婷婷国产综合精品电影 | 人人超碰91尤物精品国产| 色综合久久久久综合体| 岛国精品在线观看| 老色鬼精品视频在线观看播放| 麻豆一区二区三区| 欧美一区永久视频免费观看| 91偷拍与自偷拍精品| 午夜在线电影亚洲一区| 日韩精品一级二级| 日本免费新一区视频 | 国产精品免费aⅴ片在线观看| 亚洲欧洲日韩在线| 午夜精品久久久久影视| 色综合欧美在线| 国产成人综合亚洲网站| 视频一区视频二区在线观看| 免费视频一区二区| 国产精品123| 麻豆精品一二三| 色婷婷激情综合| 99久久精品99国产精品| 久久婷婷色综合| 亚洲国产精品成人综合| 国产成人精品午夜视频免费 | 色综合久久综合| 在线观看网站黄不卡| 精品1区2区在线观看| 亚洲午夜免费电影| 蜜桃精品视频在线观看| 成人一二三区视频| 日本韩国精品一区二区在线观看| 日韩一区二区三区精品视频| 国产精品国产三级国产aⅴ中文 | 国产一区二区三区不卡在线观看| 国产麻豆午夜三级精品| 欧美在线短视频| 国产精品免费网站在线观看| 555www色欧美视频| 国产精品高潮呻吟| 亚洲图片欧美视频| 国产精品一区二区男女羞羞无遮挡| 欧美少妇性性性| 国产欧美日韩久久| 婷婷六月综合亚洲| 色噜噜狠狠色综合中国| 国产日产精品一区| 国产精品美女久久久久久久网站| 亚洲成人自拍一区| www.视频一区| 国产成人免费高清| 国产成人啪免费观看软件| 欧美三级在线视频| 欧美精品一区二区三区很污很色的| 中文无字幕一区二区三区| 一区二区三区在线观看网站| 日韩精品最新网址| 美腿丝袜一区二区三区| 国产午夜亚洲精品不卡| 91丨九色丨蝌蚪富婆spa| 欧美高清www午色夜在线视频| 亚洲男人都懂的| 欧美另类久久久品| 亚洲一区视频在线观看视频| 国产三级欧美三级日产三级99| 欧美日韩精品一区二区天天拍小说| 亚洲精选视频在线| 91欧美一区二区| 国产日韩欧美亚洲| 成人夜色视频网站在线观看| 亚洲成人自拍一区| 精品久久久网站| 国产麻豆视频精品| 亚洲欧美日韩人成在线播放| 在线不卡免费欧美| 五月综合激情日本mⅴ| 国产女主播视频一区二区| 在线观看欧美黄色| 久久99最新地址| 亚洲欧洲av在线| 在线视频一区二区三区| 麻豆精品一区二区av白丝在线 | 久久久久久夜精品精品免费| 国产成人aaaa| 天堂va蜜桃一区二区三区| 欧美大片一区二区| 91啪亚洲精品| 亚洲aaa精品| 中文字幕精品在线不卡| 欧美性三三影院| 欧美日韩成人综合天天影院| 精品成人免费观看| 国产精品女主播av| 亚洲视频一区二区在线观看| 久久99久久99精品免视看婷婷 | 亚洲精品中文字幕乱码三区| 一本色道久久加勒比精品| 一区二区三区高清| 成人激情动漫在线观看| 中文字幕av一区二区三区| 欧美电影免费观看高清完整版在线| 丰满少妇在线播放bd日韩电影| 日本强好片久久久久久aaa| 色婷婷国产精品综合在线观看| 色婷婷香蕉在线一区二区| 国产精品日日摸夜夜摸av| 91欧美一区二区| 国产毛片一区二区|