黑马点评plus –异步秒杀重构升级

参考 阿星不是程序员 的开源项目

https://gitee.com/java-up-up/hmdp-plus

前端发起请求

localhost:8085/voucher-order/seckill/{id} , 后端接受一个voucherId 后续seckillVoucher方法执行下单逻辑

1
Result<Long> seckillVoucher(Long voucherId);

整体流程图

整流程图

查询优惠券信息(by Id)

优惠券信息查询使用多级缓存存储优惠券信息, 双重判断 + 缓存空值 + 分布式锁 + 布隆过滤器 组合的方案来解决缓存击穿和穿透的问题

流程图(gpt生成)

查询优惠券流程图

为什么要 两次检查 Redis 缓存 是否命中 ?

  • 防止缓存击穿
  • 避免大量线程同时查数据库

一个线程来到 了 ,去查 本地缓存redis 都没命中 ,就会去查 数据库,但是 此时这个线程, 还没有来得及重写到redis中时,又有一大批线程来了,此时虽然第一个线程已经查到了,但这些线程仍然需要去查数据库 ,这样就造成大量数据库的查询操作 ,影响性能 .

分布式锁+再次判断redis解决 解决缓存击穿

此时 , 一个线程来了 ,去查 本地缓存redis 都没命中, 接着 这个线程会和其他一大批线程竞争分布式锁 ,只有拿到锁的 ==线程== 继续往后走

其他线程等待 ,等这个线程执行结束了 ( 查完数据库 -> 写入缓存), 释放锁 ,其他线程才能继续竞争锁 ,往后执行, 再次判断redis 是否命中 ,命中直接返回,从而 当很多相同的并发查询来到时, 只允许一个线程去数据库查询拿信息再写回缓存(分布式锁), 大大减少了数据库的压力.

注意: 这个在查询优惠券库存时 也用到了

__END__