网站图片自动轮换怎么做的,辽宁建设工程信息网审计报告,深圳做网站排名价格,网站建设技术发展趋势预测文章目录 面试题缓存预热缓存雪崩解决方案 缓存穿透解决方案 缓存击穿解决方案案例#xff1a;高并发聚划算业务 总结表格 面试题
缓存预热、雪崩、穿透、击穿分别是什么?你遇到过那几个情况?缓存预热你是怎么做的?如何避免或者减少缓存雪崩?穿透和击穿有什么区别?他两是… 文章目录 面试题缓存预热缓存雪崩解决方案 缓存穿透解决方案 缓存击穿解决方案案例高并发聚划算业务 总结表格 面试题
缓存预热、雪崩、穿透、击穿分别是什么?你遇到过那几个情况?缓存预热你是怎么做的?如何避免或者减少缓存雪崩?穿透和击穿有什么区别?他两是一个意思还是截然不同?穿透和击穿你有什么解决方案?如何避免?假如出现了缓存不一致你有哪些修补方案?
缓存预热
PostConstruct初始化数据
缓存雪崩 大量key同时过期 解决方案
redis中key设置为永不过期 or 过期时间错开redis缓存集群实现高可用
主从哨兵 Redis Cluster 开启Redis持久化机制aof/rdb尽快恢复缓存集群
多缓存结合预防雪崩
ehcache本地缓存redis缓存
服务降级
Hystrix或者阿里sentinel限流降级
服务提供商
阿里云-云数据库Redis版
缓存穿透 先查redis无后查mysql无都查询不到该条记录但是请求每次都会打到数据库上面去导致后台数据库压力暴增 解决方案
空对象缓存或者缺省值
恶意攻击不同key打入缓存由于存在空对象缓存和缓存回写看自己业务需求redis中的无关紧要的key也会越写越多设置缓存过期时间
Google布隆过滤器Guava解决较为权威
Guava’s BloomFilter源码
白名单案例 全部合法的key都需要放入Guava版布隆过滤器redis里面不然数据就是返回nul POM
!--guava Google 开源的 Guava 中自带的布隆过滤器--
dependencygroupIdcom.google.guava/groupIdartifactIdguava/artifactIdversion23.0/version
/dependencyYML
server.port7777spring.application.nameredis# logging
logging.level.rootinfo
logging.level.com.atguigu.redis7info
logging.pattern.console%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger- %msg%n logging.file.nameredis.log
logging.pattern.file%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger- %msg%n# swagger
spring.swagger2.enabledtrue
#在springboot2.6.X结合swagger2.9.X会提示documentationPluginsBootstrapper空指针异常
#原因是在springboot2.6.X中将SpringMVC默认路径匹配策略从AntPathMatcher更改为PathPatternParser
# 导致出错解决办法是matching-strategy切换回之前ant_path_matcher
spring.mvc.pathmatch.matching-strategyant_path_matcher# redis单机
spring.redis.database0
# 修改为自己真实IP
spring.redis.host192.168.1.129
spring.redis.port6379
spring.redis.password111111
spring.redis.lettuce.pool.max-active8
spring.redis.lettuce.pool.max-wait-1ms
spring.redis.lettuce.pool.max-idle8
spring.redis.lettuce.pool.min-idle0# alibaba.druid
spring.datasource.typecom.alibaba.druid.pool.DruidDataSource
spring.datasource.driver-class-namecom.mysql.jdbc.Driver
spring.datasource.urljdbc:mysql://localhost:3306/datatest?useUnicodetruecharacterEncodingutf-8useSSLfalse
spring.datasource.usernameroot
spring.datasource.password111111
spring.datasource.druid.test-while-idlefalse# mybatis
mybatis.mapper-locationsclasspath:mapper/*.xml
mybatis.type-aliases-packagecom.atguigu.redis7.entities# redis集群
#spring.redis.password111111
## 获取失败 最大重定向次数
#spring.redis.cluster.max-redirects3
#spring.redis.lettuce.pool.max-active8
#spring.redis.lettuce.pool.max-wait-1ms
#spring.redis.lettuce.pool.max-idle8
#spring.redis.lettuce.pool.min-idle0
##支持集群拓扑动态感应刷新,自适应拓扑刷新是否使用所有可用的更新默认false关闭
#spring.redis.lettuce.cluster.refresh.adaptivetrue
##定时刷新
#spring.redis.lettuce.cluster.refresh.period2000
#spring.redis.cluster.nodes192.168.111.185:6381,192.168.111.185:6382,192.168.111.172:6383,192.168.111.172:6384,192.168.111.184:6385,192.168.111.184:6386案例
Test
public void testGuavaWithBloomFilter()
{//1 创建guava版布隆过滤器BloomFilterInteger bloomFilter BloomFilter.create(Funnels.integerFunnel(), 100);//2 判断指定的元素是否存在System.out.println(bloomFilter.mightContain(1));System.out.println(bloomFilter.mightContain(2));System.out.println();//3 讲元素新增进入bloomfilterbloomFilter.put(1);bloomFilter.put(2);System.out.println(bloomFilter.mightContain(1));System.out.println(bloomFilter.mightContain(2));}黑名单案例
缓存击穿 大量的请求同时查询一个 key 时此时这个key正好失效了就会导致大量的请求都打到数据库上面去 解决方案 常见问题热点key失效 差异失效时间对于访问频繁的热点key干脆就不设置过期时间互斥跟新采用双检加锁策略多个线程同时去查询数据库的这条数据那么我们可以在第一个查询数据的请求上使用一个 互斥锁来锁住它。其他的线程走到这一步拿不到锁就等着等第一个线程查询到了数据然后做缓存。后面的线程进来发现已经有缓存了就直接走缓存
案例高并发聚划算业务
分析 问题热点key突然失效导致缓存击穿
升级加固
双检加锁策略
多个线程同时去查询数据库的这条数据在第一个查询数据的请求上使用一个互斥锁来锁住它。
其他的线程走到这一步拿不到锁就等着等第一个线程查询到了数据然后做缓存。后面的线程进来发现已经有缓存了就直接走缓存。
差异失效时间
// Controller method
RequestMapping(value /pruduct/find,method RequestMethod.GET)
ApiOperation(聚划算案例每次1页每页5条显示)
public ListProduct find(int page, int size) {ListProduct listnull;long start (page - 1) * size;long end start size - 1;try{// 采用redis list结构里面的lrang命令来实现加载和分页查询list redisTemplate.opsForList().range(JHS_KEY,start,end);if(CollectionUtils.isEmpty(list)){//TODO 走mysql查询}log.info(参加活动的商家:{},list);}catch (Exception e){// 出异常了一般redis宕机了或者redis网络抖动导致timeoutlog.error(jhs exception:{},e);e.printStackTrace();// ....再次查询mysql}return list;
}// 模拟数据
PostConstruct
public void initJHSAB(){log.info(启动AB定时器计划任务天猫聚划算功能模拟..........DateUtil.now());//1 用线程模拟定时任务后台任务定时将mysql里面的参加活动的商品刷新到redis里new Thread(() - {while (true){//2 模拟从mysql查出数据用于加载到redis并给聚划算页面显示ListProduct list this.getProductsFromMysql();//3 先更新B缓存且让B缓存过期时间超过A缓存如果A突然失效了还有B兜底防止击穿redisTemplate.delete(JHS_KEY_B);redisTemplate.opsForList().leftPushAll(JHS_KEY_B,list);redisTemplate.expire(JHS_KEY_B,86410L,TimeUnit.SECONDS);//4 再更新A缓存redisTemplate.delete(JHS_KEY_A);redisTemplate.opsForList().leftPushAll(JHS_KEY_A,list);redisTemplate.expire(JHS_KEY_A,86400L,TimeUnit.SECONDS);//5 定时更新缓存模拟聚划算一天执行的参加活动的品牌try { TimeUnit.MINUTES.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); }}},t1).start();总结表格