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