redis电商秒杀设计

文章目录

  • redis电商秒杀设计
  • 前言
  • 一、实现方式一
  • 二、实现方式二
  • 总结

前言

在电商业务中,我们经常会遇到秒杀的业务情况,我们如何做到高成功抢购率以及实现不超卖的情况。
对于这种涉及到高并发的业务,我们通常会使用nosql去做处理。等到逻辑处理成功后才写库。


一、实现方式一

我们创建一个redis key值 为stock。库存为100。每次进入下单逻辑时,判断当前库存是否已经卖空。卖空直接return,同时设置一个是否正在处理下单逻辑的key值,如果有正在处理的订单,直接抛出异常,抢购失败。这样操作的弊端就是同一时刻只能处理一笔订单,抢购失败率会很高。

public function setStock1(){
        $redis =RedisHandler::getInstance();
        $redis->set("stock", 100);
    }

    public function placeOrder1(){
        $redis =RedisHandler::getInstance();
        $is_over = $redis->get("is_over");

        if($is_over == 1){
            echo "商品已售空";
            return;
        }
        $is_selling = $redis->get("is_selling");
        if($is_selling == 1){
            echo "抢购失败";
            return;
        }
        $redis->set("is_selling", 1);
        $new_stock = $redis->incre("stock", -1);
        echo "当前剩余库存:".$new_stock;
        sleep(1);
        if($new_stock <= 0){
            echo "商品已售空";
            $redis->set("is_over", 1);
        }else{
            //剩余库存
           $stock = $redis->get("stock");
           file_put_contents("a.txt", $stock."\r\n", FILE_APPEND);
           $redis->set("is_selling", 0);
        }
    }

二、实现方式二

使用redis队列来做。我们将需要抢购商品ID用队列的方式写入redis。然后下单时 每次重redis里面rPop弹出商品ID来进行业务处理。如果队列为空。提示抢购失败,库存卖完。这样就保证抢购成功率比较高,不会出现超卖的情况

public function setStock2(){
        $redis =RedisHandler::getInstance();
        for($i = 0; $i<100; $i++){
            $redis->rPush("stock", 1);
        }
    }

    public function placeOrder2(){
        $redis =RedisHandler::getInstance();
        $res = $redis->rPop("stock");
        if(!empty($res)){
            $stock = $redis->llen("stock");
            file_put_contents("a.txt", $stock."\r\n", FILE_APPEND);
            $redis->set("is_selling", 0);
        }else{
            echo "商品已售空";
            return;
        }
    }

总结

并不只是有上述方案可以实现秒杀业务,具体业务情况需要具体分析。
上面还有很多的附属业务的逻辑没有实现。比如支付问题,用户取消抢购成功的商品问题。