...d> 最近在做抢购系统,但头疼的是,在多用户高并发的情况下经常会库存出现问题。排查到,在同一时间内多用户同时下单导致查询和插入不同步了,而查询中跟插入又有时间差而在高并发的情况下导致库存问题(我的...

这次给大家带来php处理抢购类高并发请求实现详解,php处理抢购类高并发请求的注意事项有哪些,下面就是实战案例,一起来看一下。

本文以抢购、秒杀为例。介绍如何在高并发状况下确保数据正确。

在高并发请求下容易参数两个问题

1.数据出错,导致产品超卖。

2.频繁操作数据库,导致性能下降。

测试环境

Windows7

apache2.4.9

php5.5.12

php框架 yii2.0

工具 apache bench (apache自带高并发请求工具)。

通常处理方法

从控制器可以看出代码思路。先查询商品库存。如果库存大于0 ,则库存减少1,同时生产订单,录入抢购者数据。// 常规代码处理高并发

public function actionNormal(){

// 查询库存

$stock = Goods::find()->select('stock')->where(['goods_id'=>100001])->asArray()->one();

// 判断该商品是否还有库存

if ($stock['stock']>0) {

// 库存减一

Goods::updateAllCounters(['stock' => -1],['goods_id'=>100001]);

// 生产订单(另外功能,暂且随机赋值)

$order = $this->build_order();

// 秒杀信息入库

$model = new Highly();

$model->order_id = $order;

$model->goods_name = '秒杀商品';

$model->buy_time = date('Y-m-d H:i:s',time());

$model->mircrotime = microtime(true);

if($model->save()===false){

echo '未能成功抢购!';

}else{

echo '恭喜你,订单'.$order.'抢购成功';

}

}else{

echo '已被抢购一空!';

}

}

将商品库存设置为20后,通过ab 配置200的并发请求。ab -n 200 -c 200 http//localhost/highly/normal

执行结果发现库存变成了负值,商品超卖了。

原因比较简单,在高并发请求下。在生产订单,减少库存之前,会优先查询到库存结果。

优化一:修改库存数据类型

第一种优化方法,从数据库入手。既然查询到的结果不准确,那我就在库存减少上做手脚。将库存的数据类型改成无符号(不能有负值)。

代码还是跟上面差不多,只是在库存减1的地方做了个判断。避免报错。public function actionNormal(){

// 查询库存

$stock = Goods::find()->select('stock')->where(['goods_id'=>100001])->asArray()->one();

// 判断该商品是否还有库存

if ($stock['stock']>0) {

// 库存减一

if(Goods::updateAllCounters(['stock' => -1],['goods_id'=>100001])===false){

echo "已被抢购一空!";

return false;

}

// 生产订单(另外功能,暂且随机赋值)

$order = $this->build_order();

// 秒杀信息入库

$model = new Highly();

$model->order_id = $order;

$model->goods_name = '秒杀商品';

$model->buy_time = date('Y-m-d H:i:s',time());

$model->mircrotime = microtime(true);

if($model->save()===false){

echo '未能成功抢购!';

}else{

echo '恭喜你,订单'.$order.'抢购成功';

}

}else{

echo '已被抢购一空!';

}

}

这一次同样200的并发,执行结果发现。数据正确,并不会出现超卖的情况。

思路其实也比较简单。因为库存不能为负值,当库存等于0时,如果还有值传进来,则会报错。请求被终止。

这种优化方式,虽然避免了商品超卖的情况。但是在另一方面,请求仍然会对数据库造成压力。如果多个功能使用此数据库,会造成性能下降厉害。

优化二:redis

利用 redis list类型的pop的原子性。在操作数据库前,做一个验证。当商品卖完后,就不允许再继续进行数据库操作。// redis list 高并发测试

public function actionRedis(){

$redis = \Yii::$app->redis;

// $redis->lpush('mytest',1);

$order = $this->build_order();

// echo $order;die;

// echo $redis->llen('mytest');

$reg = $redis->lpop('mytest');

if (!$reg) {

echo "笨蛋!已经被抢光啦!";

return false;

}

$redis->close();

$model = new Highly();

$model->order_id = $order;

$model->goods_name = '秒杀商品';

$model->buy_time = date('Y-m-d H:i:s',time());

$model->mircrotime = microtime(true);

if($model->save()===false){

echo '未能成功抢购!';

}else{

echo '恭喜你,订单'.$order.'抢购成功';

}

}

// 给redis添加商品

public function actionInsertgoods(){

$count = yii::$app->request->get('count',0);

if (empty($count)) {

echo '大兄弟,你还没告诉我需要上架多少商品呢!';

return false;

}

$redis = \Yii::$app->redis;

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

$redis->lpush('mytest',1);

}

echo '成功添加了'.$redis->llen('mytest').'件商品。';

$redis->close();

}

这点的代码,我写了两个方法。第一个方法是秒杀的代码,第二个方法是给秒杀的商品设置数量。为了方便测试,我这里处理的比较简单。

通过测试,数据库生产的订单数量正常,并没有出现问题。而又避免了请求数据库造成性能下降的问题。同时内存数据库redis查询的速度要比mysql快很多。

相信看了本文案例你已经掌握了方法,更多精彩请关注php中文网其它相关文章!

推荐阅读:

php+ajax无刷新文件上传实现步骤详解

PHP实现多维数组排序算法有哪些方式

php处理抢购类功能的高并发请求,php处理抢购类相关推荐

  1. php同时抢购 代码,php如何处理抢购类功能的高并发请求

    在高并发请求下容易参数两个问题 1.数据出错,导致产品超卖. 2.频繁操作数据库,导致性能下降. 本文主要和大家详细介绍了php处理抢购类功能的高并发请求,具有一定的参考价值,感兴趣的小伙伴们可以参考 ...

  2. php 和mysql实现抢购功能_php处理抢购类功能的高并发请求

    本文以抢购.秒杀为例.介绍如何在高并发状况下确保数据正确. 在高并发请求下容易参数两个问题 1.数据出错,导致产品超卖. 2.频繁操作数据库,导致性能下降. 测试环境 Windows7 apache2 ...

  3. 【高并发】JUC底层工具类Unsafe

    1.概述 转载:添加链接描述 参考:[java]java的unsafe 参考:JUC原子类: CAS, Unsafe和原子类详解 1.1 本文主要内容 Unsafe基本介绍 获取Unsafe实例 Un ...

  4. Web大规模高并发请求和抢购的解决方案

    电商的秒杀和抢购,对我们来说,都不是一个陌生的东西.然而,从技术的角度来说,这对于Web系统是一个巨大的考验.当一个Web系统,在一秒钟内收到数以万计甚至更多请求时,系统的优化和稳定至关重要.这次我们 ...

  5. OkHttp工具类在微服高并发场景中问题实践总结

    OkHttp工具类在微服高并发场景中问题实践总结 问题场景 我的应用是一个中间业务应用XXApp,一个交易请求进来需要依赖下游应用,采用http协议通讯方式,需要调用3-4次下游请求. 老XXApp在 ...

  6. 达达O2O后台架构演进实践:从0到4000高并发请求背后的努力

    1.引言 达达创立于2014年5月,业务覆盖全国37个城市,拥有130万注册众包配送员,日均配送百万单,是全国领先的最后三公里物流配送平台. 达达的业务模式与滴滴以及Uber很相似,以众包的方式利用社 ...

  7. mysql 保证事物完整性_数据库高并发请求,如何保证数据完整性?详解MySQL/InnoDB的加锁...

    本文是对MySQL/InnoDB中,乐观锁.悲观锁.共享锁.排它锁.行锁.表锁.死锁概念的理解,这些在面试中也经常遇到,如数据库高并发请求,如何保证数据完整性?今天我查阅资料进行了MySQL/Inno ...

  8. 解秘 Node.js 单线程实现高并发请求原理,以及串联同步执行并发请求的方案

    最近在做一个支持多进程请求的 Node 服务,要支持多并发请求,而且请求要按先后顺序串联同步执行返回结果. 对,这需求就是这么奇琶,业务场景也是那么奇琶. 需求是完成了,为了对 Node.js 高并发 ...

  9. PostgreSQL数据库 OLTP高并发请求性能优化

    PostgreSQL数据库 OLTP高并发请求性能优化   2015-10-14 11:00:00|  作者:德哥:分类: PgSQL PerfTuning| 2015年度PG大象会报名地址: htt ...

最新文章

  1. 竟有如此沙雕的代码注释!
  2. SQL Server 移动数据库
  3. 微软确定 Win10 付费才能玩 还能不能愉快玩耍?
  4. PHP随机输出视频API源码,php 32行代码简单实现抖音无水印视频api,附前台页面
  5. 异步任务,邮箱任务,定时任务
  6. 基于java+swing+mysql+JFeeChart的企业人力资源管理系统(1)
  7. 如果华为完全没办法买到芯片,是否可以尝试做无芯手机?
  8. sql 查出一张表中重复的所有记录数据并且删除
  9. 知乎:学习分布式系统需要怎样的知识?
  10. Python学习笔记17:玩转千图成像
  11. 【经典】对static关键字深入理解
  12. Transformer15
  13. 用NDK-r25编译libpng
  14. 【PyTorch训练中Dataset多线程加载数据,比Dataloader里设置多个workers还要快】
  15. GNU Bison 2.1 中文手册
  16. Pytorch:RNN、LSTM、GRU 构建人名分类器(one-hot版本、Embedding嵌入层版本)
  17. SEO写作,小白如何快速写一篇高质量SEO文章
  18. 第一个PWA程序-聊天室
  19. 开发中常提到的脚手架是指的什么?
  20. C51(蓝牙,红外)智能遥控小车

热门文章

  1. 关于给hexo博客增加每日一言(诗句,影视名句,网易云热评等)
  2. Redis容量预估工具
  3. java魔法师_RxJava魔法师
  4. 长大后自卑的孩子,大多出自这几种家庭,别不当回事(给已经做父母或未来即将为人父母的你)
  5. python readcsv读取gbk编码文件_python读写csv文件
  6. Mixamo骨骼转为Unreal骨骼方法(1)
  7. Python判断节假日 chinese_calendar
  8. 国科大学习资料--人工智能原理与算法-第十二次作业解析(学长整理)
  9. 季冠携“闪星服务”受邀参加2021连锁企业轻资产论坛
  10. thymeleaf 设置时间格式