网站备案 座机号码,怎样做网站卖手机号,网站备案在外地,wordpress二次元动漫主题前提#xff1a;先把商品详情和秒杀商品缓存redis中#xff0c;减少对数据库的访问#xff08;可使用定时任务#xff09; 秒杀商品无非就是那几步#xff08;前面还可能会有一些判断#xff0c;如用户是否登录#xff0c;一人一单#xff0c;秒杀时间验证等#xff0…前提先把商品详情和秒杀商品缓存redis中减少对数据库的访问可使用定时任务 秒杀商品无非就是那几步前面还可能会有一些判断如用户是否登录一人一单秒杀时间验证等 1一人一单 2.判断库存 3.减库存 4.创建订单
秒杀需要解决的问题
1.解决超卖问题
1.1这样秒杀肯定会出现超卖的情况所以必须加锁。加锁无非就两种锁乐观锁和悲观锁。 悲观锁直接在1前上锁即可这里使用redisson里的可重入锁 // 获取分布式锁对象RLock lock redissonClient.getLock(seckillLock);try {//等待5秒获取锁执行60秒还是没有释放锁就强制释放boolean b lock.tryLock(5L, 60L, TimeUnit.SECONDS);if (b){SeckillProductVo seckillProductVo seckillProductService.find(time, seckillId);//1.保证库存足够if (seckillProductVo.getStockCount()0){return Result.error(SeckillCodeMsg.SECKILL_STOCK_OVER);}//下面两个方法必须写在一起避免事务未提交未扣减库存就释放锁//2.扣减库存//3.生成订单orderInfoorderInfoService.doSeckill(phone,seckillProductVo);}} catch (InterruptedException e) {e.printStackTrace();} finally {lock.unlock();}1.2乐观锁这个就不需要用代码演示了 1一人一单 2.判断库存 3.减库存 4.创建订单
在3.减库存的时候mysql语句 and 库存0即可MySQL会有自己的行锁。所以每次只会执行一次修改操作。以上两个方法都不会造成超卖 但是却未能解决秒杀问题原因高并发请求都进入了数据库来查库会造成数据库崩溃或阻塞其它mysql的执行。
2.重复下单问题如何解决
2.1 在4.创建订单后把用户的唯一标识id或phone存入redis的set中 key-{phone1,phone2} redisTemplate.opsForSet().isMember(orderKey,phone)返回true或false 流程变成 1.一人一单 2.判断库存 3.减库存 4.创建订单 5.redisTemplate.opsForSet().add(orderKey,phone);
2.2 但是这样并不能完全解决重复下单比如用户同时点了两次请求。极端条件下a线程进入步骤1的判断从1到4 需要进行很多操作当a线程未走到5时b线程就进来了这样也会造成重复下单。所以必须使用数据库的唯一索引进行解决把用户唯一标识和秒杀商品唯一标识做成唯一索引即可解决 也就是在 4.创建订单 的时候会出现报错3库存减掉之后数据回滚即可。 然而redis里的库存却和数据库的就不一样了redis是无法被回滚的。后面会使用cannal对数据库和redis的库存进行同步。。。 有些人可能会说了2.1没有解决重复问题为什么不直接用2.2解决2.1在大部分情况下都能解决避免有人多次点击秒杀对数据造成压力所以2.1和2.2一起使用才是最佳的选择
3.流量控制 3.1 必须把请求拦截在数据库之外如果库存只有10个也只有10个线程能最终到达数据库就是一个比较理想的方案 所以把数据库的秒杀商品库存存入redis中进行预售只有redis里能执行-1操作才能进入service这样就极大的减少了请求到service层中
3.2 如果同一个秒杀时间段有多个商品 如100个秒杀商品每个秒杀商品的库存为10个那么就会有10*100个请求到达数据库如果是淘宝的双11显然秒杀的商品会更多。那么我们就需要进行流量的削峰控制。 从controller中拦截消息service取消息的时候慢慢拿mq能够占时的让一部分请求存储在里面排队处理service根据自己的处理能力去拿。有些人可能会说排队处理不是很慢在controller中发完消息这个请求已经可以算是结束了用户不会立即看到抢购结果。后面经过 最终方案