推荐10-避免商品超卖的4种方案

原始方案(失败):在每次下订单前我们判断促销商品的数量够不够,不够不允许下订单,更改库存量时加上一个条件,只更改商品库存大于0的商品的库存,当时我们使用ab进行压力测试,当并发超过500,访问量超过2000时,还是会出现超卖现象。

public function buyOne()
{
    $shop = Shop::find(1);
    if ($shop->number > 0) {
        DB::update("update shop set number = number - 1 where id = 1");
    }
}

第1种方案:使用mysql的事务加排他锁来解决,首先我们选择数据库的存储引擎为innoDB,使用的是排他锁实现的,刚开始的时候我们测试了下共享锁,发现还是会出现超卖的现象。有个问题是,当我们进行高并发测试时,对数据库的性能影响很大,导致数据库的压力很大。

//2.利用数据库的forupdate来加锁(在数量少的情况下并不会出现问题,但是当并发达到(ab -n 1000 -c 200), //就会出现请求非2XX的响应增多,1000 失败了 60)time per request 65.195 //在高并发的情况下,会导致数据库连接数不够,部分php获取不到连接而报错,或者是超过等待时间而报错

public function indexMysql()
{
  DB::beginTransaction();
  //通过for update 加排它锁
  $shop = DB::table('shop')->where('id', '=', 1)->lockForUpdate()->first();
  if ($shop->number > 0) {
    if (DB::update("update shop set number = number - 1 where id = 1")) {
      DB::commit();
    } else {
      DB::rollBack();//回滚并重试
      usleep(100000);
      $this->indexMysql();
    }
  } else {
    DB::commit();
  }
}

第2种方案:使用文件锁实现。当用户抢到一件促销商品后先触发文件锁,防止其他用户进入,该用户抢到促销品后再解开文件锁,放其他用户进行操作。这样可以解决超卖的问题,但是会导致文件得I/O开销很大。

第3种方案:使用redis的setnx来实现锁机制。但是并发大的情况下,锁的争夺会变多,导致响应越来越慢。(与第四种方案类似)

//在数量少的情况下并不会出现问题,但是当并发达到(ab -n 1000 -c 200 就会出现请求非2XX的响应增多,1000 失败了 54) time per request 127.575

public function index()
{
  //测试并发超卖现象
  if (Redis::setnx(self::KEY, 1)) {//拿到了锁
    $this->buy();
  } else {
    usleep(100000);//等会再去拿锁
    //Log::info("未争夺到锁,睡眠100ms");
    $this->index();
  }
}
private function buy()
{
  $shop = Shop::find(1);
  if ($shop->number > 0) {
    $shop->number --;
    $shop->save();
  }
  Redis::del(self::KEY);
}

第4种方案:redis的队列来实现。将要促销的商品数量以队列的方式存入redis中,每当用户抢到一件促销商品则从队列中删除一个数据,确保商品不会超卖。这个操作起来很方便,而且效率极高

//4.使用redis队列来,用户过来直接入队列,然后再将操作更新到数据库 //最佳体验(redis pconnect 9.481s, 无丢失, 无框架)

public function push()
{
  //入队列
  Redis::lpush(self::QUEUE, 1);
}

//脚本调用pop方法 * * * * * php xxx.php

public function pop()
{
    while (($key = Redis::rpop(self::QUEUE))) {
      $shop = Shop::find(1);
      if ($shop->number > 0) {
        DB::update("update shop set number = number - 1 where id =         1")
     }
  }

}

本文分享自微信公众号 - PHP技术大全(phpgod)

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2019-09-18

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏野路子程序员

PHP读取大文件源码示例-Swoole多进程读取大文件

在日常读取文件时,若文件 不是很大,通常使用file_get_contents,将内容一次性载入的变量中,也可以远程加载网页或者远端文件。

13830
来自专栏日常杂谈

ZOL桌面壁纸的提取

这是爬虫的第一部分,对于python基础与网络编程部分重点突出,主要以每次小项目为主;更新时间不定,随缘之人,缘分到了,文章就出来了。

9810
来自专栏日常杂谈

Django中的url与视图详解(2)

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。

9020
来自专栏日常杂谈

Django中的url与视图详解(1)

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。

8830
来自专栏日常杂谈

Django中的url与视图详解(3)

可能你学习到这里,感觉好乱,所将的知识点没有一丝的关联,这个是没有办法的,Django与Flask有所不同的,Django是结构化的,每个模块都有知识点,我们只...

7520
来自专栏人工智能头条

给中级Python开发者的13个练手项目,适合你不?

该项目设计的主要目标是聚合内容。首先,我们需要知道内容聚合器从哪些站点获取内容。然后,使用请求库来发送 HTTP 请求,并使用 BeautifulSoup 解析...

11740
来自专栏腾讯云serverless的专栏

Serverless实践系列(八):如何优雅地给网站图片加水印

前言 很多论坛、博客在进行图片上传之后,都会给自己的图像加上水印,这样可以证明这张图片「属于我」或者是「来自我的博客/网站」。 传统的加水印的方法,通常是在流...

12620
来自专栏Node开发

Nginx缓存原理及机制

上篇文章介绍了Nginx一个较为重要的知识点:Nginx实现接口限流。本篇文章将介绍Nginx另一个重要知识点:Nginx缓存原理。其实说到缓存技术大家应该...

10840
来自专栏汇智网教程

Ripple区块链对接PHP开发包【瑞波币/XRP】

XrpTool可以帮助PHP应用快速接入瑞波/Ripple区块链, 即支持部署自有Ripple节点的应用场景,也支持利用公开的Ripple节点广播离线裸交易的轻...

18050
来自专栏机器学习算法与Python学习

【Python】13 个适合『中级开发者』练手的项目

该项目设计的主要目标是聚合内容。首先,我们需要知道内容聚合器从哪些站点获取内容。然后,使用请求库来发送 HTTP 请求,并使用 BeautifulSoup 解析...

11020

扫码关注云+社区

领取腾讯云代金券

年度创作总结 领取年终奖励