秒杀在商城系统里很常见,一般提到秒杀方案都是用java方案的比较多,redis+MQ方案比较多,但是php实现起来还是要麻烦一点,既然麻烦那我们就做个简单点的。
当然php也可以用MQ队列,但相对没来说,没有java那么方便,那我们就采用redis简单实现。
秒杀系统的核心功能点:
- 商家创建活动,设置时间段,选择参与活动的商品,设置折扣、库存等;
- 用户端可以看到秒杀预告列表、商品秒杀详情页用户点击立即抢购;
- 如果库存充足,则创建订单成功,否则秒杀失败。
- 提交订单后超时未支付,系统会自动关闭订单,回滚库存。
秒杀系统的核心技术难点:
- 并发流量大,秒杀开始前后,系统的瞬间请求流量飙升。
- 数据库负担过重。
- 超卖的情况。
- 不支付抢占名额,导致商品未销售出去。
thinkphp8针对以上问题做的优化方案:
目前主要用到了redis,在秒杀活动中,Redis具有以下优势:
- 高速读写:Redis基于内存,读写速度非常快,可以处理高并发的请求。
- 高可靠性:Redis支持主从复制和数据持久化,可以实现数据备份和恢复,保证数据的可靠性和一致性。
- 高并发性:Redis采用单线程模型,避免了线程切换和锁竞争的问题,可以处理大量的并发请求。
首先先把库存加到redis队列,代码如下:
$redis = new Client([
'host' => '127.0.0.1',
'port' => '6379',
'password' => '123456',
]);
$stock = 100;
$goodsKey = "goods_id_stock";
for ($i = 1; $i <= $stock; $i++) {
$redis->lpush($goodsKey, 1);
}
然后用户进行秒杀下的时候,执行出队列操作
$redis = new Client([
'host' => '127.0.0.1',
'port' => '6379',
'password' => '123456',
]);
$goodsKey = "goods_id_stock";
if($redis->lpop($goodsKey)){
//秒杀成功,处理订单,商品库存逻辑
}else{
//秒杀失败
}
另外还有其他要注意的点:
- 秒杀后不付款定时取消,回退库存等操作。这个时间可以设置短一点,以免长时间锁住库存不释放,导致回滚库存时活动已结束。
- 秒杀活动、秒杀商品等相关信息可以存redis,减少数据库负担。
- 防止重复提交,利用redis的increment操作,记录用户请求提交次数,如果是第一次秒杀请求,increment后的值肯定为1,则允许排队;如果值大于1,说明重复提交。
如何处理恶意下单请求:
- 如果限制用户只能买一件商品,把用户id和商品id作为联合主键索引
- 验证码机制
- IP限流:限制同一IP访问速率
- url加密防暴露
当然还有很多需要注意的点,要做一个完善的秒杀系统,也是花很大的时间和精力。
如果需要了解项目方案,可以在gitee搜[三勾软件],有好的建议也可以留言一起讨论。