【Redis】实现缓存及相关问题

Redis实现缓存及相关问题

认识缓存

缓存就是数据交换的缓冲区,是存贮数据的临时地方,一般读写性能较高。

缓存的作用:

  1. 降低后端负载
  2. 提高读写效率,降低响应时间

缓存的成本:

  1. 数据一致性成本
  2. 代码维护成本
  3. 运维成本

添加缓存

缓存作用模型

查询商铺缓存的流程

添加缓存业务代码

@Override
public List<UserDTO> getUserlist() {Gson gson = new Gson();// 1. 查询redis缓存String cache = redisTemplate.opsForValue().get(CACHE_LIST_PRE);// 2.1. 存在缓存if (StrUtil.isNotBlank(cache)) {// 3. 反序列化return gson.fromJson(cache, new TypeToken<List<UserDTO>>() {}.getType());}// 2.2. 不存在缓存// 3. 查询数据库List<User> userList = list();// 4. 信息脱敏ArrayList<UserDTO> userDTOList = new ArrayList<>();for (User user : userList) {UserDTO dto = new UserDTO();BeanUtil.copyProperties(user, dto);dto.setPhone(DesensitizedUtil.mobilePhone(dto.getPhone()));userDTOList.add(dto);}// 5. 保存到RedisredisTemplate.opsForValue().set(CACHE_LIST_PRE, gson.toJson(userDTOList), 2, TimeUnit.MINUTES);return userDTOList;
}

缓存更新

缓存更新策略

内存淘汰超时剔除主动更新
说明利用Redis的内存淘汰机制,内存不足时自动淘汰部分数据。给缓存数据添加TTL时间,到期后自动删除缓存。下次查询时更新缓存。编写业务逻辑,在修改数据库的同时,更新缓存。
一致性一般
维护成本
  • 低一致性需求:使用内存淘汰机制
  • 高一致性需求:主动更新 + 超时剔除

主动更新策略

  • 读操作:
    • 缓存命中则直接返回
    • 缓存未命中则查询数据库,并写入缓存,设定超时时间
  • 写操作:
    • 先操作数据库,然后再删除缓存
    • 要确保数据库与缓存操作的原子性(事物/分布式事物)

缓存穿透

什么是缓存穿透

缓存穿透是指客户端请求的数据在缓存中和数据库中都不存在,这样缓存永远不会生效,这些请求都会打到数据库。

缓存空对象

优点:实现简单,维护方便

缺点:额外的内存消耗、可能造成短期的不一致

业务实现

// 缓存空对象
@Override
public UserDTO getInfoById(Long id) {// 缓存查询String userString = redisTemplate.opsForValue().get(CACHE_USER_PRE + id);if (StrUtil.isNotBlank(userString)) {// 有缓存 => 真实数据return gson.fromJson(userString, UserDTO.class);}if (userString != null) {// 有缓存 => 空对象throw new BusinessException(404, "用户不存在");}// 数据库查询User user = getById(id);if (user == null) {// 缓存空对象redisTemplate.opsForValue().set(CACHE_USER_PRE + id, "", 2, TimeUnit.MINUTES);throw new BusinessException(404, "用户不存在");}// 信息脱敏UserDTO userDTO = BeanUtil.copyProperties(user, UserDTO.class);// 缓存真实数据redisTemplate.opsForValue().set(CACHE_USER_PRE + id, gson.toJson(userDTO), 2, TimeUnit.MINUTES);return userDTO;
}

布隆过滤器

优点:内存占用较少,没有多余key

缺点:实现复杂、存在误判可能

image-20240117202714944

缓存雪崩

缓存雪崩是指在同一时段大量的缓存key同时失效或者Redis服务宕机,导致大量请求到达数据库,带来巨大压力。

解决方案:

  1. 给不同的Key的TTL添加随机值
  2. 利用Redis集群提高服务的可用性
  3. 给缓存业务添加降级限流策略
  4. 给业务添加多级缓存

缓存击穿/热点Key

什么是缓存击穿

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

互斥锁

优点:没有额外的内存消耗、保证一致性、实现简单

缺点:线程需要等待,性能受影响、可能有死锁风险

互斥锁流程图

互斥锁业务代码

// 热点key-互斥锁
@Override
public List<UserDTO> getUserlist() throws InterruptedException {// 1. 查询redis缓存String cache = redisTemplate.opsForValue().get(CACHE_LIST_PRE);// 2.1. 存在缓存if (StrUtil.isNotBlank(cache)) {// 3. 反序列化return gson.fromJson(cache, new TypeToken<List<UserDTO>>() {}.getType());}// ⭐️ 获取互斥锁String lock = REDIS_LOCK_PRE + "userlist";Boolean flag = redisTemplate.opsForValue().setIfAbsent(lock, "1", 30, TimeUnit.SECONDS);if (BooleanUtil.isFalse(flag)) {// ⭐️ 获取锁失败了 => 休眠 + 递归Thread.sleep(200);return getUserlist();}ArrayList<UserDTO> userDTOList = new ArrayList<>();try {// 2.2. 不存在缓存// 3. 查询数据库List<User> userList = list();log.info("查询数据库");// 4. 信息脱敏for (User user : userList) {UserDTO dto = new UserDTO();BeanUtil.copyProperties(user, dto);dto.setPhone(DesensitizedUtil.mobilePhone(dto.getPhone()));userDTOList.add(dto);}// 5. 保存到RedisredisTemplate.opsForValue().set(CACHE_LIST_PRE, gson.toJson(userDTOList), 10, TimeUnit.SECONDS);} catch (Exception ignored) {} finally {// ⭐️ 释放锁redisTemplate.delete(lock);}return userDTOList;
}

逻辑过期

优点:线程无需等待,性能较好

缺点:不保证一致性、有额外内存消耗、实现复杂

逻辑过期流程图

逻辑过期业务代码

1、LogicalExpiration逻辑过期实体类,使用泛型使其通用化

@Data
@AllArgsConstructor
@NoArgsConstructor
public class LogicalExpiration<T> {private T value;private Date date;
}

2、核心业务代码

public UserDTO getUserById(Long id) {// 1. 查询缓存String userString = redisTemplate.opsForValue().get(CACHE_USER_PRE + id);if (StrUtil.isBlank(userString)) {// 没有缓存 -> 查询数据User user = getById(id);// 没有数据 -> 报错if (user == null) {throw new BusinessException(500, "用户不存在");}// 数据脱敏UserDTO userDTO = BeanUtil.copyProperties(user, UserDTO.class);// 新建缓存Date date = new Date();date.setTime(System.currentTimeMillis() + 2 * 60 * 1000);redisTemplate.opsForValue().set(CACHE_USER_PRE + id, gson.toJson(new LogicalExpiration<>(userDTO, date)));// 返回数据return userDTO;}// 存在缓存 => 反序列化拿到对象LogicalExpiration<UserDTO> logicalExpiration = gson.fromJson(userString, new TypeToken<LogicalExpiration<UserDTO>>() {}.getType());// 判断缓存是否过期if (logicalExpiration.getDate().before(new Date())) {// 已经过期 => 新建线程进行更新缓存ExecutorService executorService = Executors.newSingleThreadExecutor();executorService.execute(() -> {UserDTO userDTO = BeanUtil.copyProperties(getById(id), UserDTO.class);Date date = new Date();// 设置TTL为2mindate.setTime(System.currentTimeMillis() + 2 * 60 * 1000);redisTemplate.opsForValue().set(CACHE_USER_PRE + id, gson.toJson(new LogicalExpiration<>(userDTO, date)));});}// 返回数据return logicalExpiration.getValue();
}

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

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

相关文章

vue学习91-105

vue的基本认知p91 创建一个空仓库p93 vue 路由 vuex版本 2 3 3 3 4 4 npm的vuex装包npm install vuex --save vuex里有仓库,仓库放vuex核心代码&#xff0c;所有组件都能访问到 const store new Vuex.Store()//访问stored this.$store如何提供$访问vuex的数据p94 核心概念-…

梁航影院影视混剪,手机影视教学+电脑影视教学

课程下载&#xff1a;https://download.csdn.net/download/m0_66047725/88804173 手机影视教学&#xff1a; 发布作品的时间及发布过程中注意的问题 剪辑思路 如何参与变现 十六比九视频制作教程 账号定位及思路提升 作品的垂直度以及如何选剧找剧 作品质量提升及案例实…

【爬虫实战】全过程详细讲解如何使用python获取抖音评论,包括二级评论

简介&#xff1a; 前两天&#xff0c;TaoTao发布了一篇关于“获取抖音评论”的文章。但是之前的那一篇包涵的代码呢仅仅只能获取一级评论。虽然说抖音的一级评论挺精彩的了&#xff0c;但是其实二级评论更加有意思&#xff0c;同时二级评论的数量是很多。所以二级评论是非常值…

Go语言Gin框架安全加固:全面解析SQL注入、XSS与CSRF的解决方案

前言 在使用 Gin 框架处理前端请求数据时&#xff0c;必须关注安全性问题&#xff0c;以防范常见的攻击。本文将探讨 Gin 框架中常见的安全问题&#xff0c;并提供相应的处理方法&#xff0c;以确保应用程序的稳健性和安全性。 处理前端请求数据时&#xff0c;确保应用程序的…

【智慧工业】东胜物联定位与跟踪解决方案,为方案商提供蓝牙网关、信标等物联网智能硬件设备

利用东胜物联的蓝牙网关我们的合作伙伴在德国的建筑工地成功实施了基于物联网蓝牙的员工出勤和跟踪管理解决方案&#xff0c;该解决方案简化了员工时间表并增强了工作流程&#xff0c;为经理和主管提供了更多时间来专注于项目洞察&#xff0c;并提高了员工的效率、绩效和生产力…

【Shell的运行原理以及Linux当中的权限问题】

Shell的运行原理以及Linux当中的权限问题 Shell的运行原理Linux当中的权限问题Linux权限的概念如何实现用户账号之间的切换如何仅提升当前指令的权限如何将普通用户添加到信任列表 Linux权限管理文件访问者的分类 (人)文件类型和访问权限 (事物属性)文件权限值的表示方法文件访…

java入门、环境配置及其特点介绍

目录 一、java语言的重要特点 二、java开发工具包&#xff08;JDK&#xff09;及其环境配置 三、java入门代码 四、Java运行机制 五、java学习方法 一、java语言的重要特点 java是面向对象的Java是健壮性的。Java具有强类型机制、异常处理、垃圾的自动收集等特点java语言是跨…

前端登陆加密解决方案

项目背景 环食药烟草的数据下载模块中&#xff0c;需要判断用户在进行数据下载时是进行了登录操作&#xff0c;如果没有登录要跳转登陆页面&#xff0c;输入账号和密码进行登录。 使用场景 项目中需要前端书写登录页面&#xff0c;用户输入账号密码&#xff0c;前端获取到用…

protoc结合go完成protocol buffers协议的序列化与反序列化

下载protoc编译器 下载 https://github.com/protocolbuffers/protobuf/releases ps: 根据平台选择需要的编译器&#xff0c;这里选择windows 解压 加入环境变量 安装go专用protoc生成器 https://blog.csdn.net/qq_36940806/article/details/135017748?spm1001.2014.3001.…

React 中实现拖拽功能-插件 react-beautiful-dnd

拖拽功能在平时开发中是很常见的&#xff0c;这篇文章主要使用react-beautiful-dnd插件实现此功能。 非常好用&#xff0c;附上GitHub地址&#xff1a;https://github.com/atlassian/react-beautiful-dnd 安装及引入 // 1.引入 # yarn yarn add react-beautiful-dnd# npm npm…

SDL库的下载与配置(Visual Studio )2024/2/4更新

一.SDL的下载 下载链接 二.SDL的环境配置 解压以后放在中文路径下 不会添加环境变量自行搜索&#xff08;比较简单网上教程很多&#xff09; 下面进行编译器的配置 复制这段内容 x64\SDL2main.lib x64\SDL2.lib将这段代码放进去运行一下 #include <SDL.h>int main(int…

MySQL 架构和性能优化

重点&#xff1a; 视图&#xff0c;函数&#xff0c;存储过程&#xff0c;触发器&#xff0c;事件&#xff08; 了解 &#xff09; 用户管理&#xff0c;密码管理 grant revoke 权限管理 MySQL 架构&#xff08; 了解 &#xff09; 存储引擎&#xff1a;MyISAM 和 InnoDB …

Vscode配置STM32开发环境(联合Keil MDK/IAR开发)

Vscode配置STM32开发环境&#xff08;替代Keil MDK/IAR&#xff09; 前言 使用了很长时间的Keil5 MDK&#xff0c;以及最近用了一段时间的IAR for ARM&#xff0c;总体来说体验都不是特别的好&#xff0c;Keil功能还行&#xff0c;也不卡顿&#xff0c;就是开发效率、界面样式…

【蓝桥杯冲冲冲】[NOIP2001 普及组] 装箱问题

蓝桥杯备赛 | 洛谷做题打卡day26 文章目录 蓝桥杯备赛 | 洛谷做题打卡day26题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 提示思路 题解代码我的一些话 [NOIP2001 普及组] 装箱问题 题目描述 有一个箱子容量为 V V V&#xff0c;同时有 n n n 个物品&#xff0c;每…

echarts使用之柱状图(一)

1 基本使用 核心配置主要是xAxis/yAxis/series <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><meta http-equi…

2018年苏州大学837复试机试C/C++

2018年苏州大学复试机试 要求 要求用C/C编程&#xff1b;对程序中必要的地方进行注释。上机规则 请在电脑桌面上新建一个文件夹文件夹名为考试姓名&#xff08;中文&#xff09;&#xff1b;考试完毕后&#xff0c;将所编写的文件放在上述文件中。 第一题&#xff08;20分&…

07、全文检索 -- Solr -- Solr 全文检索 之 为索引库添加中文分词器

目录 Solr 全文检索 之 为索引库添加中文分词器添加中文分词器1、添加中文分词器的 jar 包2、修改 managed-schema 配置文件什么是 fieldType 3、添加 停用词文档4、重启 solr5、添加【*_cn】动态字段&#xff0c;并为该字段设置中文分词器6、演示分词器的区别演示 text_cjk 这…

瑞_23种设计模式_建造者模式

文章目录 1 建造者模式&#xff08;Builder Pattern&#xff09;1.1 介绍1.2 概述1.3 创作者模式的结构 2 案例一2.1 需求2.2 代码实现 3 案例二3.1 需求3.2 代码实现 4 模式拓展 ★★★4.1 重构前4.2 重构后 5 总结5.1 建造者模式优缺点5.2 建造者模式使用场景5.3 建造者模式 …

scikit-learn 1.3.X 版本 bug - F1 分数计算错误

如果您正在使用 scikit-learn 1.3.X 版本&#xff0c;在使用 f1_score() 或 classification_report() 函数时&#xff0c;如果参数设置为 zero_division1.0 或 zero_divisionnp.nan&#xff0c;那么函数的输出结果可能会出错。错误的范围可能高达 100%&#xff0c;具体取决于数…

STM32单片机的C语言基础

C语言是单片机开发中的必备基础知识&#xff0c;本文列举了部分STM32学习中比较常见的一些C语言基础知识。 1 位操作 下面我们先讲解几种位操作符&#xff0c;然后讲解位操作使用技巧。C语言支持以下六种位操作&#xff1a; 下面&#xff0c;重点讲解一下位操作在单片机开发中…