php处理抢购类功能的高并发请求
作者:春风不及你的笑 发布时间:2023-11-21 01:30:54
本文以抢购、秒杀为例。介绍如何在高并发状况下确保数据正确。
在高并发请求下容易参数两个问题
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 '恭喜你,订单<b>'.$order.'</b>抢购成功';
}
}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 '恭喜你,订单<b>'.$order.'</b>抢购成功';
}
}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 '恭喜你,订单<b>'.$order.'</b>抢购成功';
}
}
// 给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快很多。
来源:http://blog.csdn.net/u011061889/article/details/77417739
猜你喜欢
- 原作者:Jonathan 翻译:charlee原文:http://f6design.com/journal/2006/10/21/the-v
- 数据安全是任何数据服务解决方案中的一个关键要求,而Windows Server 2008和SQL Server 2008结合起来,通过一个基
- 学习python爬虫时遇到了一个问题,书上有示例如下:import reline='Cats are smarter than do
- 缓动,学名为Tween,缓冲移动的简称。要想页面内容切换起来舒服,就使用淡入淡出特效,要想让页面元素动起来自然,就要使用缓动效果。这两个混合
- JSP 开发之 releaseSession的实例详解Hibernate可以实现分页查询,昨天试了一下,分页效果不错。但是发现了一个问题,就
- Socket服务器是网络服务中常用的服务器。使用go语言实现这个业务场景是很容易的。这样的网络通讯,需要一个服务端和至少一个客户端。我们计划
- Go 中时间格式化的模板const ( ANSIC = "Mon Jan _2 15:04:
- 原文地址:30 Days of Mootools 1.2 Tutorials - Day 4 - Functions函数和MooTools
- Python3 解释器Linux/Unix的系统上,一般默认的 python 版本为 2.x,我们可以将 python3.x 安装在 /us
- 一、实现代码1.sql-- phpMyAdmin SQL Dump-- version 4.5.1-- http://www.phpmyad
- 一、join函数(一)参数使用说明描述Python join() 方法用于将序列中的元素以指定的字符连接生成一个新的字符串。语法join()
- 代码如下:create table A_TEST ( PAYOUT_ITEM_CODE VARCHAR2(30) not null, FOR
- 代码如下:USE [tempdb] GO /****** Object: UserDefinedFunction [dbo].[fun_ge
- SCRIPT 标记 用于包含JavaScript代码. 属性 LANGUAGE&nbs
- test.asp 测试演示文件clsrsa.asp 实现rsa加密与解密的vbs类文件下面是代码:1. test.asp<%rem 文
- 分页,就是按照某种规则显示分组数据集,但是在SQL Server 中,分页并不是十分容易就能够实现。在过去,开发人员通常需要自己编写程序,使
- 想要一个这玩意,可是找了网上许多着色器,要么是兼容性成问题,要么是匹配不精确,比如说:1、注释里包含字符串、关键词,类似于:/* xxxx&
- 有感于中国互联网设计界十几年的变化,从网页设计师变身界面设计师,和近一两年来兴起的转型交互设计师。大多数都是随着一个行业的兴起而前赴后继的投
- 如何在刷新链接之前验证文件是否存在?如何在每次刷新链接之前,验证链接文件是否确实存在?特别是对于数据库中的文件,我们必须在处理它之前确认一下
- 首先要说明的是,这个标题有点标题党的意思,这个 bug 也存在于 IE8 下,在 IE6 和 IE7 下正常。之前写过两篇关于 I