Redis消息隊(duì)列實(shí)現(xiàn)秒殺教程
Redis消息隊(duì)列實(shí)現(xiàn)秒殺
消息隊(duì)列的應(yīng)用場(chǎng)景例如:秒殺、搶單功能。
下面寫(xiě)個(gè)Demo簡(jiǎn)單實(shí)現(xiàn)一下秒殺,也就是搶購(gòu)。
首先創(chuàng)建一個(gè)lpush.html文件
代碼如下:
<!DOCTYPE html> <html> <head> <title>TODO supply a title</title> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <script src="jquery-3.2.1.min.js" ></script> <script> function push(uid){ $('#msg'+uid).html('請(qǐng)耐心等待.......'); $.get('lpush.php',{"uid":uid},function (data){ if(data == 1 ){ $('#msg'+uid).html('搶購(gòu)成功!!'); }else{ $('#msg'+uid).html('搶購(gòu)失敗!!'); } }); } </script> </head> <body> <span id="msg5"></span><br> <input type="button" value="搶購(gòu)5" onclick="push(5)"><br> <span id="msg15"></span><br> <input type="button" value="搶購(gòu)15" onclick="push(15)"><br> <span id="msg25"></span><br> <input type="button" value="搶購(gòu)25" onclick="push(25)"><br> <span id="msg35"></span><br> <input type="button" value="搶購(gòu)35" onclick="push(35)"><br> <span id="msg45"></span><br> <input type="button" value="搶購(gòu)45" onclick="push(45)"><br> <span id="msg55"></span><br> <input type="button" value="搶購(gòu)55" onclick="push(55)"><br> </body> </html>
在創(chuàng)建一個(gè)lpush.php文件用來(lái)處理數(shù)據(jù)
這個(gè)文件里頭的內(nèi)容呢就是執(zhí)行入隊(duì)操作,代碼如下:
<?php //實(shí)例化redis對(duì)象 $redis = new redis(); //連接redis,第一個(gè)參數(shù)是redis服務(wù)的IP127.0.0.1是自己的,6379是端口號(hào) $redis->connect('127.0.0.1', 6379); //接收ajax傳來(lái)的值,請(qǐng)看lpush.html $uid = $_GET['uid']; //入隊(duì)列 $redis->lpush('tv', $uid); //寫(xiě)一個(gè)死循環(huán)用來(lái)判斷結(jié)果 while (true) { //守護(hù)進(jìn)程是否將所有訂單處理完畢 if ($redis->scard('tvuser') == 5) { //判斷當(dāng)前用戶是否有對(duì)應(yīng)的訂單 if ($redis->sismember('tvuser' , $uid)) { $flag = 1; } else { $flag = 0; } //返回給頁(yè)面 echo $flag; //滿足條件后停止循環(huán) break; } } ?>
在創(chuàng)建一個(gè)守護(hù)進(jìn)程的文件pop.php
這個(gè)文件里頭的主要內(nèi)容就是去隊(duì)列里頭的值并且返回人數(shù)是否已夠,代碼如下:
<?php //實(shí)例化redis對(duì)象 $redis = new redis(); //連接redis,第一個(gè)參數(shù)是redis服務(wù)的IP127.0.0.1是自己的,6379是端口號(hào) $redis->connect('127.0.0.1', 6379); //定義一個(gè)變量用來(lái)存儲(chǔ)循環(huán)次數(shù) $num = 0; //循環(huán)取值 while (true) { //每隔一秒取一次 sleep(1); //redis取值 $uid = $redis->rpop('tv'); //判斷取出來(lái)的UID又沒(méi)有值 if ($uid > 0) { //有值 //create order 生成用戶對(duì)應(yīng)的訂單 echo 'order created for user ' . $uid . "\r\n"; $redis->sadd('tvuser' , $uid); //搶購(gòu)成功的用戶id //如果有值的話num會(huì)++ $num++; } else { //無(wú)值 //insert log echo 'no uid ' . "\r\n"; } //如果num值大于等于5那么循環(huán)終止,循環(huán)次數(shù)是看秒殺多少件商品來(lái)變的,這里的話我就定義了5次 if ($num >= 5) { break; } } ?>
我們?cè)趫?zhí)行守護(hù)進(jìn)程之前可以先把redis里的數(shù)據(jù)清空一下,以免擾亂程序執(zhí)行結(jié)果,
命令如下:
效果就是這個(gè)樣子的,在秒殺開(kāi)始之前要先啟動(dòng)守護(hù)進(jìn)程,
命令效果如下:
然后守護(hù)進(jìn)程啟動(dòng)之后,前臺(tái)就可以來(lái)進(jìn)行秒殺了。
瀏覽器上訪問(wèn)你的lpush.html,由于我們后臺(tái)設(shè)定的商品數(shù)為5,那么只能五個(gè)用戶搶購(gòu)成功,第6個(gè)用戶就會(huì)提示失敗,如下圖:
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Redis瞬時(shí)高并發(fā)秒殺方案總結(jié)
本文講述了Redis瞬時(shí)高并發(fā)秒殺方案總結(jié),具有很好的參考價(jià)值,感興趣的小伙伴們可以參考一下,具體如下:2018-05-05Redis數(shù)據(jù)庫(kù)的數(shù)據(jù)傾斜詳解
Redis,英文全稱(chēng)是Remote Dictionary Server(遠(yuǎn)程字典服務(wù)),是一個(gè)開(kāi)源的使用ANSI C語(yǔ)言編寫(xiě)、支持網(wǎng)絡(luò)、可基于內(nèi)存亦可持久化的日志型、Key-Value數(shù)據(jù)庫(kù),需要的朋友可以參考下2023-07-07Redis教程(十五):C語(yǔ)言連接操作代碼實(shí)例
這篇文章主要介紹了Redis教程(十五):C語(yǔ)言連接操作代碼實(shí)例,本篇博客是該系列博客中的最后一篇,在這里將給出基于Redis客戶端組件訪問(wèn)并操作Redis服務(wù)器的代碼示例,需要的朋友可以參考下2015-05-05redis哨兵模式分布式鎖實(shí)現(xiàn)與實(shí)踐方式(redisson)
這篇文章主要介紹了redis哨兵模式分布式鎖實(shí)現(xiàn)與實(shí)踐方式(redisson),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03利用Redis實(shí)現(xiàn)訂單30分鐘自動(dòng)取消
本文主要介紹了利用Redis實(shí)現(xiàn)訂單30分鐘自動(dòng)取消,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06Redis教程(六):Sorted-Sets數(shù)據(jù)類(lèi)型
這篇文章主要介紹了Redis教程(六):Sorted-Sets數(shù)據(jù)類(lèi)型,本文講解了Sorted-Sets數(shù)據(jù)類(lèi)型概述、相關(guān)命令列表、命令使用示例、應(yīng)用范圍等內(nèi)容,需要的朋友可以參考下2015-04-04