Thinkphp5+Redis实现商品秒杀代码实例讲解


Posted in PHP onDecember 29, 2020

环境:wamp,redis

要求:安装WAMP,Redis,以及为PHP安装Redis扩展

秒杀功能大致思路:获取缓存列表的长度,如果长度(llen)等于0,就停止秒杀,即秒杀失败,如果长度大于0,则继续运行,先从缓存中移除一个元素(lpop),再进行数据库操作(添加订单表,商品库存数量减一),如果再进一个人秒杀,就再走一遍流程,循环往复。

一、安装Redis扩展

1.查看PHP版本信息

打开phpinfo.php,查看PHP版本,我的是PHP7.3.4,还有一个需要注意Architecture x64

Thinkphp5+Redis实现商品秒杀代码实例讲解

2.下载扩展文件

https://pecl.php.net/package/redis

https://pecl.php.net/package/igbinary

根据自己环境,选择合适的版本

3.解压

解压下载的压缩包,并把php_redis.dll、php_redis.pdb和php_igbinary.dll、php_igbinary.pdb四个文件,移至自己PHP版本对应目录下的ext文件夹下E:\phpstudy_pro\Extensions\php\php7.3.4nts\ext

Thinkphp5+Redis实现商品秒杀代码实例讲解

Thinkphp5+Redis实现商品秒杀代码实例讲解

4.修改php.ini

添加如下代码:

extension=php_igbinary.dll
extension=php_redis.dll

如果有这两句可以把前面的分号删掉,没有就自己添加上,要注意顺序,php_igbinary.dll 要在php_redis.dll 前面

Thinkphp5+Redis实现商品秒杀代码实例讲解

5.重启Apache

重启后,再运行phpinfo.php,查看是否安装成功

Thinkphp5+Redis实现商品秒杀代码实例讲解

二、数据结构

一共三张表,ab_goods商品表,ab_order订单表,ab_log日志表

商品表

Thinkphp5+Redis实现商品秒杀代码实例讲解

订单表

Thinkphp5+Redis实现商品秒杀代码实例讲解

日志表 记录秒杀信息

Thinkphp5+Redis实现商品秒杀代码实例讲解

三、代码

<?php
namespace app\index\controller;
use think\Controller;
use think\Db;
use think\cache\driver\Redis;

class Miaosha extends Controller
{

 private $redis = null;
 private $cachekey = null; //缓存变量名
 private $basket = []; //私有数组,存放商品信息

 private $store = 50;

 /**
 * 购物车初始化,传入用户id
 */
 public function __construct()
 {
 parent::__construct();

 $this->redis = new \Redis(); // 实例化
 $this->redis->connect('127.0.0.1','6379');
 $this->redis->auth('zxf123456');

 }

 /**
 * 秒杀初始化
 */
 public function Ms_init()
 {
 // 删除缓存列表
 $this->redis->del($this->cachekey);

 $len = $this->redis->llen($this->cachekey);
 $count = $this->store - $len;

 for ($i=0; $i < $count; $i++) { 

 // 向库存列表推进50个,模拟50个商品库存
 $this->redis->lpush($this->cachekey,1);
 }

 echo "库存初始化完成:".$this->redis->llen($this->cachekey);
 }
 

 /**
 * 秒杀入口
 */
 public function index()
 {
 $id = 1; //商品编号
 
 if (empty($id)) {
 // 记录失败日志
 return $this->writeLog(0,'商品编号不存在'); 
 }

 // 计算库存列表长度
 $count = $this->redis->llen($this->cachekey);

 // 先判断库存是否为0,为0秒杀失败,不为0,则进行先移除一个元素,再进行数据库操作
 if ($count == 0) { //库存为0

 $this->writeLog(0,'库存为0');
 echo "库存为0";
 exit;

 }else{
 // 有库存
 //先移除一个列表元素
 $this->redis->lpop($this->cachekey);

 $ordersn = $this->build_order_no(); //生成订单
 $uid = rand(0,9999); //随机生成用户id
 $status = 1;
 // 再进行数据库操作
 $data = Db::table('ab_goods')->field('count,amount')->where('id',$id)->find(); //查找商品

 if (!$data) {
 return $this->writeLog(0,'该商品不存在');
 }

 $insert_data = [
 'order_sn' => $ordersn,
 'user_id' => $uid,
 'goods_id' => $id,
 'price' => $data['amount'],
 'status' => $status,
 'addtime' => date('Y-m-d H:i:s')
 ];

 // 订单入库
 $result = Db::table('ab_order')->insert($insert_data);
 // 自动减少一个库存
 $res = Db::table('ab_goods')->where('id',$id)->setDec('count');

 if ($res) {
 echo "第".$count."件秒杀成功";
 $this->writeLog(1,'秒杀成功');
 }else{
 echo "第".$count."件秒杀失败";
 $this->writeLog(0,'秒杀失败');
 }
 }
 }

 /**
 * 生成订单号
 */
 public function build_order_no()
 {
 return date('ymd').substr(implode(NULL, array_map('ord', str_split(substr(uniqid(), 7, 13), 1))), 0, 8);
 }

 /**
 * 生成日志 1成功 0失败
 */
 public function writeLog($status = 1,$msg)
 {
 $data['count'] = 1;
 $data['status'] = $status;
 $data['addtime'] = date('Y-m-d H:i:s');
 $data['msg'] = $msg;
 return Db::table('ab_log')->insertGetId($data);
 }

}

四、压力测试

使用apache压力测试工具 AB 测试,模拟多用户秒杀商品,模拟60秒内发起3000个请求,并发600次,秒杀50个库存商品

AB测试相关参数说明

  • -r 指定接收到错误信息时不退出程序
  • -t 等待响应的最大时间
  • -n 指定压力测试总共的执行次数
  • -c 用于指定压力测试的并发数

1.初始化50个库存,运行ms_init方法

2.测试   命令行:

E:\phpstudy_pro\Extensions\Apache2.4.39\bin>ab -r -t 60 -n 3000 -c 1000 http://gouwuche.zxf/index/miaosha/index  

Thinkphp5+Redis实现商品秒杀代码实例讲解

3.检测数据库数据

Thinkphp5+Redis实现商品秒杀代码实例讲解

Thinkphp5+Redis实现商品秒杀代码实例讲解

日志表状态为1(秒杀成功)的数据有50人,订单表里的订单数也是50条,商品表里的商品数量变成了0(测试之前是50),商品秒杀成功完成!

如果不用redis而是直接用mysql的话,商品表订单的数量count会变成负数,而秒杀成功的人数也多余50人,订单表里的订单数量也多余50条(新测),下面是直接用Mysql的例子;

public function sqlMs()
 {
 $id = 1; //商品编号

 $count = 50;
 $ordersn = $this->build_order_no(); //生成订单
 $uid = rand(0,9999); //随机生成用户id
 $status = 1;
 // 再进行数据库操作
 $data = Db::table('ab_goods')->field('count,amount')->where('id',$id)->find(); //查找商品

 // 查询还剩多少库存
 $rs = Db::table('ab_goods')->where('id',$id)->value('count');
 if ($rs <= 0) {
 
 $this->writeLog(0,'库存为0');
 }else{

 $insert_data = [
 'order_sn' => $ordersn,
 'user_id' => $uid,
 'goods_id' => $id,
 'price' => $data['amount'],
 'status' => $status,
 'addtime' => date('Y-m-d H:i:s')
 ];

 // 订单入库
 $result = Db::table('ab_order')->insert($insert_data);
 // 自动减少一个库存
 $res = Db::table('ab_goods')->where('id',$id)->setDec('count');

 if ($res) {
 echo "第".$data['count']."件秒杀成功";
 $this->writeLog(1,'秒杀成功');
 }else{
 echo "第".$data['count']."件秒杀失败";
 $this->writeLog(0,'秒杀失败');
 }
 }
 }

到此这篇关于Thinkphp5+Redis实现商品秒杀的文章就介绍到这了,更多相关Thinkphp5+Redis实现商品秒杀内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

PHP 相关文章推荐
php知道与问问的采集插件代码
Oct 12 PHP
php curl 登录163邮箱并抓取邮箱好友列表的代码(经测试)
Apr 07 PHP
php数据结构与算法(PHP描述) 查找与二分法查找
Jun 21 PHP
自定义php类(查找/修改)xml文档
Mar 26 PHP
php验证是否是md5编码的简单代码
Apr 01 PHP
php通过function_exists检测函数是否存在的方法
Mar 18 PHP
PHP中Closure类的使用方法及详解
Oct 09 PHP
PHP页面输出时js设置input框的选中值
Sep 30 PHP
DWZ+ThinkPHP开发时遇到的问题分析
Dec 12 PHP
PHP多进程编程之僵尸进程问题的理解
Oct 15 PHP
php成功操作redis cluster集群的实例教程
Jan 13 PHP
PhpStorm2020 + phpstudyV8 +XDebug的教程详解
Sep 17 PHP
PHP序列化和反序列化深度剖析实例讲解
Dec 29 #PHP
PHP实现简单注册登录系统
Dec 28 #PHP
php的lavarel框架中join和orWhere的用法
Dec 28 #PHP
php中yar框架实例用法讲解
Dec 27 #PHP
php中数组最简单的使用方法
Dec 27 #PHP
用Laravel轻松处理千万级数据的方法实现
Dec 25 #PHP
PHP操作Redis常用命令的实例详解
Dec 23 #PHP
You might like
snoopy PHP版的网络客户端提供本地下载
2008/04/15 PHP
php strcmp使用说明
2010/04/22 PHP
PHP中json_encode、json_decode与serialize、unserialize的性能测试分析
2010/06/09 PHP
jquery pagination插件实现无刷新分页代码
2009/10/13 Javascript
javascript贪吃蛇完整版(源码)
2013/12/09 Javascript
同域jQuery(跨)iframe操作DOM(实例讲解)
2013/12/19 Javascript
Jquery获得控件值的三种方法总结
2014/02/13 Javascript
Javascript解析URL方法详解
2014/12/05 Javascript
使用JavaScript脚本判断页面是否在微信中被打开
2016/03/06 Javascript
ECMAScript6快速入手攻略
2016/07/18 Javascript
详解Python中logging日志模块在多进程环境下的使用
2016/12/26 Javascript
基于vue2.0+vuex的日期选择组件功能实现
2017/03/13 Javascript
微信小程序视图控件与bindtap之间的问题的解决
2019/04/08 Javascript
KnockoutJS数组比较算法实例详解
2019/11/25 Javascript
[02:20]DOTA2亚洲邀请赛 EHOME战队出场宣传片
2015/02/07 DOTA
[00:12]DAC2018 天才少年转战三号位,他的SOLO是否仍如昔日般强大?
2018/04/06 DOTA
python计算时间差的方法
2015/05/20 Python
使用Mixin设计模式进行Python编程的方法讲解
2016/06/21 Python
Python中将字典转换为列表的方法
2016/09/21 Python
Python入门必须知道的11个知识点
2018/03/21 Python
pandas把dataframe转成Series,改变列中值的类型方法
2018/04/10 Python
Python框架Flask的基本数据库操作方法分析
2018/07/13 Python
Python中单线程、多线程和多进程的效率对比实验实例
2019/05/14 Python
Python 的AES加密与解密实现
2019/07/09 Python
Django自定义全局403、404、500错误页面的示例代码
2020/03/08 Python
x-ua-compatible content=”IE=7, IE=9″意思理解
2013/07/22 HTML / CSS
详解HTML5 Canvas绘制不规则图形时的非零环绕原则
2016/03/21 HTML / CSS
法国设计制造的扫帚和刷子:Andrée Jardin
2018/12/06 全球购物
教师自荐书
2013/10/08 职场文书
20年同学聚会邀请函
2014/02/04 职场文书
项目施工员岗位职责
2014/03/09 职场文书
经销商年会策划方案
2014/05/29 职场文书
导游词之岳阳楼
2019/09/25 职场文书
浅谈mysql返回Boolean类型的几种情况
2021/06/04 MySQL
javascript函数式编程基础
2021/09/15 Javascript
Python爬取奶茶店数据分析哪家最好喝以及性价比
2022/09/23 Python