秒杀的理解

项目场景:

秒杀限时或者限量的处理一件商品


实现理解:

秒杀业务流程
如果使用java逻辑判断减库存的话,会出现多个线程同时修改好库存,然后存入相同的修改值
实际上是卖出了多个 但是数据库只扣了一个库存,首先前台传进来商品ID,减库存(使用sql语句的stock=stock-1  这个会触发mysql的行级锁 适用于高并发的场景)。

秒杀的理解_第1张图片
 

 秒杀的高并发实现:利用Redis,来作为一级缓存。把商品id和库存放到redis里。使用stringRedisTemple.opsForValue().set(商品ID号,库存);设置进redis。然后,秒杀下单的时候传入商品ID号,自动递减redis里面的商品库存。stringRedisTemple.opsForValue().decrement(id号); 但是,这样会出现问题。

比如:下单异常的时候会出现redis和mysql扣的库存不一致。解决方法就是在处理异常的地方添加还原redis库存数量。stringRedisTemple.opsForValue().increment(id号);但是,这样还有可能在高并发时出现少买的情况,解决就是在比对redis里面的库存,判断库存小于0的时候,也添加还原库存的代码。

这样问题差不多都解决了。但是每次都要判断如果刚好最后一个进入多个线程,同样也要判断好多次,浪费资源,这时候可以使用jvm缓存同步。使用JUC标记。

private static concurrentHashMap StockOut=new concurrentHashMap<>();

 在判断库存小于0的逻辑代码内,标记该商品号已售出为true.然后在进入这个判断之前直接判断该标记。这样提升了单体应用秒杀时候的并发量。如果是分布式的服务的话,因为每个服务的jvm不一样,无法同步在异常处理的时候还原该标记。我们可以使用队列(RabbitMq),来实现分布式服务的同步修改JVM标记。实现秒杀的性能。

你可能感兴趣的:(java,redis,开发语言,缓存,rabbitmq)