【Redis】缓存击穿的产生情况解决方案

news/2024/5/10 7:09:03/文章来源:https://blog.csdn.net/liuwanqing233333/article/details/128131028

1. 缓存击穿产生

也叫做 热点 Key 问题,高并发访问并且缓存重建业务较复杂的 key 突然失效了,无数的请求想要重建缓存,大量的访问会在瞬间给数据库带来巨大冲击。

在这里插入图片描述

2. 解决方案

2.1 方案一:互斥锁

查询缓存不存在时,为查询数据库重构缓存这一操作加一互斥锁,使得多线程并发下,只有一个线程去查询数据库,其他线程等待,减小对数据库的压力。

在这里插入图片描述

优点:

  • 无额外内存消耗
  • 能保证一致性
  • 实现起来比较简单

缺点:

  • 进程需要等待,性能受到影响:所有请求都必须等待第一个线程完成缓存重构。
  • 可能会有死锁的风险:可能我获取了一个锁了,另一个锁在另一个业务那里,但是另一个业务也等待着我释放锁,我们两个互相等待对方释放,导致死锁。

互斥锁的代码实现:

在这里插入图片描述

使用 setnx 命令获取锁,使用 del 释放锁 ,setnx 命令对应 StringRedisTemplate 中的 opsForValue().setIfAbsent ,设置成功返回 true,设置失败返回 fasle

public static final String CACHE_SHOPSTYPE_LIST = "cache:shopsType:";@Resourceprivate StringRedisTemplate stringRedisTemplate; // 注入// 获取锁的方法private boolean tryLock(String key){boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, "1", 10, TimeUnit.SECONDS);return BooleanUtil.isTrue(flag); // 防止出现 null}// 释放锁的方法private void unlock(String key){stringRedisTemplate.delete(key);}// 解决缓存击穿public  Shop queryWithMutex(Long id)  {String key = CACHE_SHOP_KEY  + id;// 1. 从 Redis 中查询商户缓存String shonJson = stringRedisTemplate.opsForValue().get(key);        // 2. 判断是否存在// 2. 命中真实值,返回数据if(!StringUtil.isNullOrEmpty(shonJson)){Shop shop = JSONUtil.toBean(shonJson, Shop.class);return shop;}// 3. 命中空字符串,返回null (缓存穿透的添加空值解决方案)if(shonJson != null){return null;}// 4. 实现缓存重建try {// 4.1. 获取互斥锁boolean isLock = tryLock("lock:shop:" + id);// 4.2. 获取失败,等待一段时间,再去请求缓存 —— 使用递归实现if(!isLock){Thread.sleep(50);return queryWithMutex(id);}// 4.3. 获取成功,重建缓存Thread.sleep(200); // 模拟重建延迟// 4.3.1. 查询数据库shop = getById(id);// 4.3.1.1 数据库不存在 —— 将空值写入缓存(缓存穿透解决方案),返回 nullif(shop == null){stringRedisTemplate.opsForValue().set(key, "", 2L, TimeUnit.MINUTES);return null;}// 4.3.1.2 数据库存在 —— 写入 Redis ,然后返回,实现缓存重建stringRedisTemplate.opsForValue().set(key, JSONUtil.toJsonStr(shop), 30L, TimeUnit.MINUTES);} catch (InterruptedException e) {throw new RuntimeException(e);} finally {// 11. 缓存重构结束,释放锁unlock("lock:shop:" + id);}return shop;}

2.2 方案二:逻辑过期

不设置 TTL ,为 Value 加上 expire 逻辑过期时间,通过业务逻辑判断缓存是否过期。

这样线程去查询缓存,其会判断逻辑时间是否过期,过期了会另开一个新线程,去更新数据。本线程不等待,直接返回过期的数据。若同时,另一个线程也来查询,发现得不到互斥锁,就会直接返回过期的数据,而不会等待。

在这里插入图片描述

在这里插入图片描述

优点:

  • 线程无需等待,性能好

缺点:

  • 不保证一致性
  • 有额外内存消耗
  • 实现复杂,需要进行逻辑判断

逻辑过期的代码实现

在这里插入图片描述

  1. 把对象封装为 RedisData ,在 RedisData 中添加 LocalDateTime 属性
package com.hmdp.utils;
import lombok.Data;
import java.time.LocalDateTime;@Data
public class RedisData { // 把对象封装为 RedisData ,存入 Redis 中private LocalDateTime expireTime;private Object data;
}
  1. 逻辑过期关键代码实现

解释: 使用线程池开启新线程执行缓存重构,其他的线程直接返回过期数据。

    // 创建一线程池private  static final ExecutorService CACHE_REBUILD_EXECUTOR = Executors.newFixedThreadPool(10);// 获取锁的方法private boolean tryLock(String key){boolean flag = stringRedisTemplate.opsForValue().setIfAbsent(key, "1", 10, TimeUnit.SECONDS);return BooleanUtil.isTrue(flag); // 防止出现 null}// 释放锁的方法private void unlock(String key){stringRedisTemplate.delete(key);}// 缓存重构方法,为对象加入 expireSeconds 逻辑过期时间 // 传递的指为要缓存的对象 id ,和缓存的时间public void saveShop2Redis(Long id, Long expireSeconds) throws InterruptedException {Shop shop;shop = getById(id);Thread.sleep(200);if (shop == null){log.info("为 null");}RedisData redisData = new RedisData();redisData.setData(shop);redisData.setExpireTime(LocalDateTime.now().plusSeconds(expireSeconds));stringRedisTemplate.opsForValue().set(CACHE_SHOP_KEY+id, JSONUtil.toJsonStr(redisData));}// 使用逻辑过期的方式解决缓存击穿public  Shop queryWithLogicalExpire(Long id)  {String key = CACHE_SHOP_KEY  + id;// 1. 从 Redis 中查询商户缓存String shonJson = stringRedisTemplate.opsForValue().get(key);        // 2. 判断是否存在// 2. 未命中if(StringUtil.isNullOrEmpty(shonJson)){return null;}// 3. 命中// 3.1 得到缓存对象 —— JSON 反序列化为 RedisData 对象RedisData redisData = JSONUtil.toBean(shonJson, RedisData.class);JSONObject data = (JSONObject)redisData.getData();Shop shop = JSONUtil.toBean(data, Shop.class);LocalDateTime expireTime = redisData.getExpireTime(); // 得到过期时间// 3.2 判断是否过期if(expireTime.isAfter(LocalDateTime.now())){// 3.2.1 未过期,直接返回return shop;}// 3.2.1 过期,需要进行缓存重建// 3.2.1.1 获取互斥锁String lockKey = "lock:shop:" + id; // 锁的 keyboolean isLock = tryLock(lockKey);// 3.2.1.2 判断是否成功获取到互斥锁// 3.2.1.2.1 获取互斥锁失败,直接返回过期数据if(!isLock){return shop;}// 3.2.1.2.2 获取互斥锁成功,在开启一个线程进行缓存重建,然后返回数据CACHE_REBUILD_EXECUTOR.submit(()->{try {// 重建saveShop2Redis(id, 20L); // 设置 30 分钟} catch (Exception e) {throw new RuntimeException(e);} finally {// 释放锁unlock(lockKey);}});return shop;}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_228898.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

天宇优配|上架秒光 “3时代”的大额存单受宠

“最近理财产品动摇比较大,准备处理一笔大额存单,但查询发现,某国有行暂时没有可购买的大额存单产品。”11月29日,成都市民王女士向金融出资报记者表示。 记者发现,虽然通过数次下调,中长期大额存单利率已步…

k8s网络插件之Calico

Calico简介 Calico官方文档:https://projectcalico.docs.tigera.io/getting-started/kubernetes/quickstart Calico是一套开源的网络和网络安全解决方案,用于容器、虚拟机、宿主机之前的网络连接,它是一个纯三层的虚拟化网络解决方案&#…

MYSQL中AS(取别名)

文章目录0 写在前面1 格式2 举例2.1 设置表别名2.2 设置字段别名3 写在末尾0 写在前面 在做业务,在mybatis中手写sql中再多表查询去映射实体时,总会用到AS这个关键字。 或者我们在数据库大量字段测试数据时,很多字段都有相同的前缀&#xff…

神仙级编程神器,吹爆

Visual Studio 编程领域公认的“最强IDE”,Visual Studio是目前最流行的Windows平台应用程序的集成开发环境,提供了高级开发工具、调试功能、数据库功能和创新功能,帮助在各种平台上快速创建当前最先进的应用程序,开发新的程序。 …

借助cubeMX实现STM32MP157A(-M4核)UART、按键中断、环境检测开关实验

main.c 可以添加一句打印提示 int main(void) {/* USER CODE BEGIN 1 *//* USER CODE END 1 *//* MCU Configuration--------------------------------------------------------*//* Reset of all peripherals, Initializes the Flash interface and the Systick. */HAL_Init(…

【内网安全】——Linux信息收集

作者名:Demo不是emo 主页面链接:主页传送门 创作初心:舞台再大,你不上台,永远是观众,没人会关心你努不努力,摔的痛不痛,他们只会看你最后站在什么位置,然后羡慕或鄙夷座…

行业应用之无限可能,就在亚马逊云科技re:Invent

在2022亚马逊云科技re:Invent全球大会Adam Selipsky“如何借助云的力量,在未知领域抓住机遇并茁壮成长”的主题演讲中,除了阐述主要的产品升级以外,亚马逊云科技还致力于打造面向特定行业或者特定应用场景的解决方案,以帮助客户快…

【react-笔记】

目录简介基本使用虚拟dom的两种创建方法jsx语法规则模块与组件、模块化和组件化的理解模块组件模块化组件化函数式组件类式组件组件实例三大属性statepropsrefs事件处理包含表单的组件分类非受控组件受控组件高阶函数_函数的柯里化生命周期引出生命周期理解生命周期(旧)总结新的…

基于jsp+mysql+ssm协同办公系统-计算机毕业设计

项目介绍 本公司文档协同办公管理系统采用SSM(SpringSpringMVCMyBatis)框架开发,主要包括系统用户管理模块、用户信息模块、文件信息管理、个人事务管理、资料信息管理、登录模块、和退出模块等多个模块. 本系统主要包含了等系统用户管理、用户信息管理…

数据结构学习笔记(Ⅶ):查找

目录 1 查找 1.1 定义 1.2 查找操作 1.3 算法评价指标 2 查找算法 2.1 顺序查找 1.算法思想 2.实现 3.查找效率 4.算法优化 2.2 折半查找 1.算法思想 2.算法实现 3.查找判定树 4.折半查找效率 2.3 分块查找 1.算法思想 2.查找效率分析 3 B树 3.1 B树概念 3…

独立站SaaS系统站群模式怎么玩

做独立站的人都知道“站群”这个游戏,意思是通过建站工具一次性建好几百个或者几千个独立站。各个独立站卖的品类比较垂直,不会有太多SKU。销量好的站会留着精细化运营,没流量的就放弃。 使用脸书或谷歌和其他广告渠道来测试产品。每个产品…

手把手教你写Linux线程池

手把手教你写Linux线程池 如果需要线程池源码,关注Linux兵工厂,并由大量Linux资料赠送。 线程池 顾名思义,存储线程的池子。线程池是线程的一种使用模式。在平常业务开发中常规的逻辑是遇到任务然后创建线程去执行。但是线程的频繁创建就类…

Java Tomcat内存马——Listener内存马

目录 (一)前置知识 0x01 什么是Listener 0x02 Listener的简单案例 0x03 Listener流程分析 (二)注入分析 (三)实现内存马 得到完整的内存马 (四)漏洞复现 其他的payload: 总结 (一&#…

Java+JSP+MySQL基于SSM的雷锋车队管理系统的设计与实现-计算机毕业设计

项目介绍 随着我国国民经济的发展和人文素质的不断提高,越来越多的爱心人士出现在了社会的各种角落之中,其中的哥和爱心人士,组织了一种基于交通和车辆之间的互助的民间组织,这种组织叫做雷锋爱心车队,而且雷锋爱心车…

什么牌子蓝牙耳机通话质量好?通话质量好的蓝牙耳机推荐

蓝牙耳机作为手机的最佳伴侣,已经成为老百姓日常生活必备。每次有大品牌发布新款蓝牙耳机,几乎都能够得到很好的反响,蓝牙耳机不仅在音质上有了很大的提升,并且在其他功能也在不断的提升,使用蓝牙耳机通话避免不了电话…

商务部研究院信用所、启信宝联合发布《中国商务信用发展指数报告(2022)》

近期,商务部国际贸易经济合作研究院信用研究所与合合信息全资子公司上海生腾数据科技有限公司(简称“生腾数据”)联合发布了《中国商务信用发展指数报告(2022)》(简称《报告》)。为准确反映中国…

glxy_阿里云存储

阿里云OSS储存 讲师的添加实现:oss服务 访问并登陆阿里云,,实名认证 产品分类---->对象储存OSS 开通OSS 进入管理控制台 使用OSS前先创建bucket java 代码实现 准备工作:创建操作阿里云oss许可证(阿里云颁发…

得一微冲刺科创板上市:拟募资约12亿元,2021年营收同比增长260%

撰稿|汤汤 来源|贝多财经 近日,得一微电子股份有限公司(下称“得一微”)在上海证券交易所科创板递交招股书(申报稿)。本次冲刺科创板上市,得一微拟公开发行不超过2354万股股份,计划募资12.24亿…

zookeeper学习(一)zk特性与节点数据类型详解(2022)

Zookeeper是一个开源的分布式协调框架,主要用来解决分布式集群中应用系统的一致性问题。从设计模式角度来理解其实zk是一个基于观察者模式设计的分布式服务管理框架。 CAP理论: cap理论指出对于一个分布式计算系统来说,不可能同时满足以下三…

golang知识点整理

目录 1、goroutine GMP模型 2、goroutine阻塞的处理 3、goroutine内存泄漏 4、map原理、扩容 5、go内存管理 6、go的gc 1、goroutine GMP模型 1. G代表一个goroutine对象,每次go调用的时候,都会创建一个G对象 2. M代表一个线程,每次创建…