侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

php和redis实现秒杀活动的流程

Redis  /  管理员 发布于 5年前   306

1 说明

前段时间面试的时候,一直被问到如何设计一个秒杀活动,但是无奈没有此方面的实际经验,所以只好凭着自己的理解和一些资料去设计这么一个程序

主要利用到了redis的string和set,string主要是利用它的k-v结构去对库存进行处理,也可以用list的数据结构来处理商品的库存,set则用来确保用户进行重复的提交

其中我们最主要解决的问题是

-防止并发产生超抢/超卖

2 流程设计

3 代码

3.1 服务端代码

class MiaoSha{ const MSG_REPEAT_USER = '请勿重复参与'; const MSG_EMPTY_STOCK = '库存不足'; const MSG_KEY_NOT_EXIST = 'key不存在'; const IP_POOL = 'ip_pool'; const USER_POOL = 'user_pool'; /** @var Redis */ public $redis; public $key; public function __construct($key = '') {  $this->checkKey($key);  $this->redis = new Redis(); //todo 连接池  $this->redis->connect('127.0.0.1'); } public function checkKey($key = '') {  if(!$key) {   throw new Exception(self::MSG_KEY_NOT_EXIST);  } else {   $this->key = $key;  } } public function setStock($value = 0) {  if($this->redis->exists($this->key) == 0) {   $this->redis->set($this->key,$value);  } } public function checkIp($ip = 0) {  $sKey = $this->key . self::IP_POOL;  if(!$ip || $this->redis->sIsMember($sKey,$ip)) {   throw new Exception(self::MSG_REPEAT_USER);  } } public function checkUser($user = 0) {  $sKey = $this->key . self::USER_POOL;  if(!$user || $this->redis->sIsMember($sKey,$user)) {   throw new Exception(self::MSG_REPEAT_USER);  } } public function checkStock($user = 0, $ip = 0) {  $num = $this->redis->decr($this->key);  if($num < 0 ) {   throw new Exception(self::MSG_EMPTY_STOCK);  } else {   $this->redis->sAdd($this->key . self::USER_POOL, $user);   $this->redis->sAdd($this->key . self::IP_POOL, $ip);   //todo add to mysql   echo 'success' . PHP_EOL;   error_log('success' . $user . PHP_EOL,3,'/var/www/html/demo/log/debug.log');  } } /**  * @note:此种做法不能防止并发  * @func checkStockFail  * @param int $user  * @param int $ip  * @throws Exception  */ public function checkStockFail($user = 0,$ip = 0) {  $num = $this->redis->get($this->key);  if($num > 0 ){   $this->redis->sAdd($this->key . self::USER_POOL, $user);   $this->redis->sAdd($this->key . self::IP_POOL, $ip);   //todo add to mysql   echo 'success' . PHP_EOL;   error_log('success' . $user . PHP_EOL,3,'/var/www/html/demo/log/debug.log');   $num--;   $this->redis->set($this->key,$num);  } else {   throw new Exception(self::MSG_EMPTY_STOCK);  } }}

3.2 客户端测试代码

function test(){ try{  $key = 'cup_';  $handler = new MiaoSha($key);  $handler->setStock(10);  $user = rand(1,10000);  $ip = $user;  $handler->checkIp($ip);  $handler->checkUser($user);  $handler->checkStock($user,$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->setStock(10);  $user = rand(1,10000);  $ip = $user;  $handler->checkIp($ip);  $handler->checkUser($user);  $handler->checkStockFail($user,$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.04
  • redis2.8.4
  • php5.5

在服务端代码里面我们有两个函数分别是checkStock和checkStockFail,其中checkStockFail不能在高并发的情况下效果很差,不能在redis层面保证库存为0的时候终止操作。

我们利用ab工具进行测试

其中 www.hello.com 是配置的虚拟主机名称 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的值,因此会产生超卖的情况

以上所述是小编给大家介绍的php和redis实现秒杀活动的流程,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

您可能感兴趣的文章:

  • php+redis消息队列实现抢购功能
  • php结合redis实现高并发下的抢购、秒杀功能的实例
  • PHP实现Redis单据锁以及防止并发重复写入
  • PHP使用Redis实现防止大并发下二次写入的方法
  • PHP商品秒杀问题解决方案实例详解【mysql与redis】
  • php+redis实现商城秒杀功能
  • yii框架redis结合php实现秒杀效果(实例代码)
  • php+redis实现消息队列功能示例
  • PHP+redis实现的限制抢购防止商品超发功能详解


  • 上一条:
    PHP商品秒杀问题解决方案实例详解【mysql与redis】
    下一条:
    PHP+redis实现微博的拉模型案例详解
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 在Redis中能实现的功能、常见应用介绍(0个评论)
    • 2024年Redis面试题之一(0个评论)
    • 在redis缓存常见出错及解决方案(0个评论)
    • 在redis中三种特殊数据类型:地理位置、基数(cardinality)估计、位图(Bitmap)使用场景介绍浅析(2个评论)
    • Redis 删除 key用 del 和 unlink 有啥区别?(1个评论)
    • 近期文章
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2017-12
    • 2020-03
    • 2020-05
    • 2021-04
    • 2022-03
    • 2022-05
    • 2022-08
    • 2023-02
    • 2023-04
    • 2023-07
    • 2024-01
    • 2024-02
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客