基于php和redis如何设计一个秒杀系统?
大概的要求是:能处理高并发请求,抢购成功但是在规定时间内未支付的订单需要从新回到系统。
我自己想一个大概思路:分为两部分 1.先由Api随机踢掉一部分用户,让少部分流量打到后段业务;2.由于秒杀,所以必须要要控制超卖,建议使用Redis队列预先把商品写入队列中,到点使用key去pop,如果pop出有,则视为抢到,这样可以防止超卖。长期未为购买的用户,可以使用cron来跑,或者把生成的订单号,放到redis里面去,结算的时候踢掉,然后使用cron进行辅助。或者可以使用beanstalk,延时任务队列。最好都还是用cron来辅助。大概思路就这样。
<?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;
}
}
?>
2 回答1.3k 阅读✓ 已解决
1 回答955 阅读✓ 已解决
2 回答839 阅读✓ 已解决
1 回答1k 阅读✓ 已解决
2 回答895 阅读
2 回答837 阅读
1 回答883 阅读
秒杀可以考虑使用redis队列来实现。
对于30分钟未支付的订单有两种方案可考虑:
第一种方案:被动过期+cron,就是用户查看的时候去数据库查有没有支付+定时清理。
第二种方案:延迟性任务,到时间检查订单是否支付成功,如果没有支付则取消订单。