基于php和redis如何设计一个秒杀系统?

基于php和redis如何设计一个秒杀系统?

大概的要求是:能处理高并发请求,抢购成功但是在规定时间内未支付的订单需要从新回到系统。

阅读 5.5k
4 个回答

秒杀可以考虑使用redis队列来实现。

对于30分钟未支付的订单有两种方案可考虑:

第一种方案:被动过期+cron,就是用户查看的时候去数据库查有没有支付+定时清理。
第二种方案:延迟性任务,到时间检查订单是否支付成功,如果没有支付则取消订单。

我自己想一个大概思路:分为两部分 1.先由Api随机踢掉一部分用户,让少部分流量打到后段业务;2.由于秒杀,所以必须要要控制超卖,建议使用Redis队列预先把商品写入队列中,到点使用key去pop,如果pop出有,则视为抢到,这样可以防止超卖。长期未为购买的用户,可以使用cron来跑,或者把生成的订单号,放到redis里面去,结算的时候踢掉,然后使用cron进行辅助。或者可以使用beanstalk,延时任务队列。最好都还是用cron来辅助。大概思路就这样。

Redis有一个list的数据结构可以满足你的需求,详情可以看一下慕课网的教程高峰削流

<?php
$redis = new redis();
$result = $redis->connect('127.0.0.1', 6379);
$mywatchkey = $redis->get("mywatchkey");
$rob_total = 100;   //抢购数量
if($mywatchkey<$rob_total){
    $redis->watch("mywatchkey");
    $redis->multi();
    
    //设置延迟,方便测试效果。
    sleep(5);
    //插入抢购数据
    $redis->hSet("mywatchlist","user_id_".mt_rand(1, 9999),time());
    $redis->set("mywatchkey",$mywatchkey+1);
    $rob_result = $redis->exec();
    if($rob_result){
        $mywatchlist = $redis->hGetAll("mywatchlist");
        echo "抢购成功!<br/>";
        echo "剩余数量:".($rob_total-$mywatchkey-1)."<br/>";
        echo "用户列表:<pre>";
        var_dump($mywatchlist);
    }else{
        echo "手气不好,再抢购!";exit;
    }
}
?>
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题