1 说明
前段时间面试的时候,一直被问到如何设计一个秒杀活动,但是无奈没有此方面的实际经验,所以只好凭着自己的理解和一些资料去设计这么一个程序
主要利用到了redis的string和t,string主要是利用它的k-v结构去对库存进行处理,也可以用list的数据结构来处理商品的库存,t则用来确保用户进行重复的提交
其中我们最主要解决的问题是
-防止并发产生超抢/超卖
2 流程设计
3 代码
3.1 服务端代码
class miaosha{ const msg_repeat_ur = '请勿重复参与'; const msg_empty_stock = '库存不足'; const msg_key_not_exist = 'key不存在'; const ip_pool = 'ip_pool'; const ur_pool = 'ur_pool'; /** @var redis */ public $redis; public $key; public function __construct($key = '') { $this->checkkey($key); $this->redis = new redis(); //todo 连接池 $this->redis->conn云南大学呈贡校区ect('127.0.0.1'); } public function checkkey($key = '') { if(!$key) { throw new exception(lf::msg_key_not_exist); } el { $this->key = $key; } } public function tstock($value = 0) { i耶鲁大学音乐学院f($this->redis->exists($this->key) == 0) { $this->redis->t($this->key,$value); } } public function checkip($ip = 0) { $skey = $this->key . lf::ip_pool; if(!$ip || $this->redis->sismember($skey,$ip)) { throw new exception(lf::msg_repeat_ur); } } public function checkur($ur = 0) { $skey = $this->key . lf::ur_pool; if(!$ur || $this->redis->sismember($skey,$ur)) { throw new exception(lf::msg_repeat_ur); } } public function checkstock($ur = 0, $ip = 0) { $num = $this->redis->decr($this->key); if($num < 0 ) { throw new exception(lf::msg_empty_stock); } el { $this->redis->sadd($this->key . lf::ur_pool, $ur); $this->redis->sadd($this->key . lf::ip_pool, $ip); //todo add to mysql echo 'success' . php_eol; error_log('success' . $ur . php_eol,3,'/var/www/html/demo/log/debug.log'); } } /** * @note:此种做法不能防止并发 * @func checkstockfail * @param int $ur * @param int $ip * @throws exception */ public function checkstockfail($ur = 0,$ip = 0) { $num = $this->redis->get($this->key); if($num > 0 ){ $this->redis->sadd($this->key . lf::ur_pool, $ur); $this->redis->sadd($this->key . lf::ip_pool, $ip); //todo add to mysql echo 'success' . dreamboardphp_eol; error_log('success' . $ur . php_eol,3,'/var/www/html/demo/log/debug.log'); $num--; $this->redis->t($this->key,$num); } el { throw new exception(lf::msg_empty_stock); } }}
3.2 客户端测试代码
function test(){ try{ $key = 'cup_'; $handler = new miaosha($key); $handler->tstock(10); $ur = rand(1,10000); $ip = $ur; $handler->checkip($ip); $handler->checkur($ur); $handler->checkstock($ur,$ip); } catch (\exception $e) { echo $e->getmessage() . php_eol; error_log('fail' . $e->getmessage() .php_eol,3,'/var/www/html/demo/log/debug.log'); }}function test2(金得利快餐){ try{ $key = 'cup_'; $handler = new miaosha($key); $handler->tstock(10); $ur = rand(1,10上篮技巧30招000); $ip = $ur; $handler->checkip($ip); $handler->checkur($ur); $handler->checkstockfail($ur,$ip); //不能防止并发的 } catch (\exception $e) { echo $e->getmessage() . php_eol; error_log('fail' . $e->getmessage() .php_eol,3,'/var/www/html/demo/log/debug.log'); }}
4 测试
测试环境说明
ubantu16.04redis2.8.4php5.5在服务端代码里面我们有两个函数分别是checkstock和checkstockfail,其中checkstockfail不能在高并发的情况下效果很差,不能在redis层面保证库存为0的时候终止操作。
我们利用ab工具进行测试
其中 是配置的虚拟主机名称 flash-sale.php
是我们脚本的名称
#第1种情况 500并发下 用客户端的test2()去执行 ab -n 500 -c 100 www.hello.com/flash-sale.php
log日志的记录结果:
#第2种情况 5000并发下 用客户端的test2()去执行 ab -n 5000 -c 1000 www.hello.com/flash-sale.php
log日志的记录结果:
#第3种情况 500并发下 用客户端的test()去执行 ab -n 500 -c 100 www.hello.com/flash-sale.php
log日志的记录结果:
#第4种情况 5000并发下 用客户端的test()去执行 ab -n 5000 -c 1000 www.hello.com/flash-sale.php
log日志的记录结果:
5 总结
我们从日志中可以很明显的看出第3、4中情况下,可以保证商品的数量总是我们设置的库存值10,但是在情况1、2下,则产生了超卖的现象
redis来控制并发主要是利用了其api都是原子性操作的优势,从checkstock和checkstockfail中可以看出,一个是直接decr对库存进行减一操作,所以不存在并发的情况,但是另一个方法是将库存值先取出做减一操作然后再重新赋值,这样的话,在并发下,多个进程会读取到多个库存为1的值,因此会产生超卖的情况
以上所述是www.887551.com给大家介绍的php和redis实现秒杀活动的流程,希望对大家有所帮助
本文发布于:2023-04-07 14:50:29,感谢您对本站的认可!
本文链接:https://www.wtabcd.cn/fanwen/zuowen/9a32c0dc5d5a93051b28bcbc34b15c89.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文word下载地址:php和redis实现秒杀活动的流程.doc
本文 PDF 下载地址:php和redis实现秒杀活动的流程.pdf
留言与评论(共有 0 条评论) |