如何用Spring Cache实现对Redis的抽象

我们在进行Java项目开发时候,经常会用到Redis缓存例如数据库里的一些信息、手机验证码之类的,正常写法就会像去连mysql一样,这种硬编码的方式肯定是非常不合适的。

  @Autowireprivate UserMapper userMapper;@Autowireprivate StringCommand stringCommand;//查询用户public User getUserById(Long userId) {String cacheKey = "userId_" + userId;User user=stringCommand.get(cacheKey);if(user != null) {return user;}user = userMapper.getUserById(userId);if(user != null) {stringCommand.set(cacheKey,user);return user;}//修改用户public void updateUser(User user){userMapper.updateUser(user);String cacheKey = "userId_" + userId.getId();stringCommand.set(cacheKey , user);}//删除用户public void deleteUserById(Long userId){userMapper.deleteUserById(userId);String cacheKey = "userId_" + userId.getId();stringCommand.del(cacheKey);}}

有没有啥更抽象、更优雅的方式来实现呢?

有!那就是使用Spring Cache,它是一个对缓存的抽象,即使使用的缓存不是Redis也可以。
在这里插入图片描述
我们看到这里最重要的是这个CacheManager,是他给我们提供了一个抽象,让我们不用关心底层缓存技术(如 Redis、Ehcache、Caffeine)

那如何使用呢?
当然是通过Spring中最常见的方式——注释了 !!!这也是CacheManager提供给我们的功能
在这里插入图片描述
在pom文件下引入对应包(spring-boot-starter-data-redis、spring-boot-starter-cache)后,在启动类上加@EnableCaching注释,就可以使用Spring Cache的功能了。

这里再解释一下注释里的value和key分别是啥:

  • 首先value指定的是缓存的名称,对应底层 Redis 的 key 前缀或 hash 名,也就是你缓存数据时存在哪个“区域”。比如下面这个示例,缓存内容都是存入名为 userCache 的缓存区。
  • key则是指定缓存项在缓存区中的唯一标识。如果不指定,Spring 会默认根据方法所有参数自动生成一个 key。
// 先查缓存,有则返回缓存内容;如果没有,执行方法,并把返回值放进缓存。
@Cacheable(value = "userCache", key = "#id")
public User getUserById(Long id) {// 方法体只会在缓存未命中时执行return userService.getUserFromDB(id);
}// 还有一种写法,假设查询对应id用户为空时,这样可以让查询结果不缓存
// unless表示条件成立时,不缓存,也就是说:如果方法返回结果是 null,则不缓存
@Cacheable(value = "userCache", key = "#id", unless = "#result == null")
public User getUserById(Long id) {// 方法体只会在缓存未命中时执行return userService.getUserFromDB(id);
}// 不查缓存,始终执行方法,然后将方法返回值更新到缓存中。
@CachePut(value = "userCache", key = "#user.id")
public User updateUser(User user) {// 更新数据库return userRepository.save(user);
}// 删除缓存中的数据,可以指定 key,也可以一次清空整个缓存。
@CacheEvict(value = "userCache", key = "#id")
// @CacheEvict(value = "userCache", allEntries = true) 这个是全部清空
public void deleteUser(Long id) {// 删除数据库中的数据userRepository.deleteById(id);
}

还需要强调的一点,可能你会知道RedisTemplate,在我另一篇文章也讲了,这是一种给你手动操作Redis数据的方式,你可以调用.opsForXXX()来获取数据,是一种更加灵活、精细操作Redis的方式,例如在实现分布式锁等场景可以采用。但我们本文介绍的CacheManager是通过注释的方式自动操作缓存的,是一种方法级的自动缓存,简洁优雅,推荐用于 Controller/Service 层。

来个更加易懂的方式来解释,RedisTemplate像是你手动操作数据库的 JDBC 接口,而CacheManager则像是用 ORM 框架(比如 MyBatis)配置好后,通过注解自动执行数据库操作。

@Configuration
@EnableCaching // 开启Spring Cache注解支持,这样的话就可以直接使用@@Cacheable、@CachePut、@CacheEvict等注释来用了,当然如果你已经在主程序上标了,这里就不用写了,@EnableCaching全局写一次就行
public class RedisConfig {@Beanpublic RedisCacheManager redisCacheManager(RedisConnectionFactory factory, CacheTtlProperties ttlProps) {// 创建默认缓存配置:设置序列化方式、key前缀、是否缓存null值、默认过期时间等RedisCacheConfiguration defaultConfig = RedisCacheConfiguration.defaultCacheConfig().serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(new StringRedisSerializer())) // key 使用字符串序列化.serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(new GenericJackson2JsonRedisSerializer())); // value 使用 JSON 序列化// 创建一个 map 存放不同缓存区域(cacheName)对应的 TTL(过期时间)配置Map<String, RedisCacheConfiguration> cacheConfigurations = new HashMap<>();// 从application.yml配置中读取每个缓存区域自定义的 TTL,并生成对应的配置项for (Map.Entry<String, Long> entry : ttlProps.getTtl().entrySet()) {cacheConfigurations.put(entry.getKey(),defaultConfig.entryTtl(Duration.ofMillis(entry.getValue()))); // 将该 cacheName 设置为自定义 TTL}// 构建 RedisCacheManager:指定默认配置及每个缓存区域的个性化配置return RedisCacheManager.builder(factory).cacheDefaults(defaultConfig) // 设置默认缓存行为.withInitialCacheConfigurations(cacheConfigurations) // 设置个性化缓存行为.build(); // 构建缓存管理器}
}

然后记得在application.yml配置对应参数,例如过期时间等:

spring:redis:host: localhost # Redis 服务地址port: 6379       # Redis 端口号cache:type: redis      # 使用 Redis 作为缓存实现redis:time-to-live: 1800000 # 全局默认缓存过期时间(单位:毫秒,1800000ms = 30分钟)cache-null-values: false # 是否缓存 null,false 表示不缓存 null 值,假设想在每个地方自行判断要不要缓存null可以不写这行use-key-prefix: true     # 是否启用 key 前缀key-prefix: cache::      # 缓存 key 的统一前缀,也就是key都变成了cache::xxx
// 上面配置的是全局缓存过期时间,你如果还想单独指定redis缓存空间例如userCache、orderCache的过期时间可以写在下面,其余的空间仍然是30分钟
custom:cache:ttl:userCache: 600000     # userCache 缓存区域的 TTL(单位:毫秒,600000ms = 10分钟)orderCache: 3600000   # orderCache 缓存区域的 TTL(单位:毫秒,3600000ms = 1小时)

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

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

相关文章

CMake指令:file()

目录 1.简介 2.常用子命令&#xff08;COMMAND&#xff09; 2.1.COPY - 复制文件或目录 2.2.RENAME - 重命名文件或目录 2.3.REMOVE - 删除文件或目录 2.4.MAKE_DIRECTORY - 创建目录 2.5.READ - 读取文件内容 2.6.WRITE - 写入文件内容 2.7.GLOB - 按模式匹配文件 2…

使用VuePress开发日志

结合官方教程&#xff0c;补充一些细节。 快速上手 | VuePress中文文档 | VuePress中文网 VuePress使用步骤 创建并进入一个新目录 mkdir vuepress-starter && cd vuepress-starter使用你喜欢的包管理器进行初始化 yarn init # npm init将 VuePress 安装为本地依赖 …

随手记录7

2025年5月26日~2025年6月01日 周一&#xff1a;没做 周二&#xff1a;芹菜炒鸡蛋香肠 周三&#xff1a; 周四&#xff1a; 周五&#xff1a; 周六&#xff1a; 周日&#xff1a;

【无标题】使用JEasyOpc开发OPCDA采集中间件

使用JEasyOpc开发OPCDA采集中间件 1.JEasyOpc下载2.修改JEasyOpc源码及打包安装3.Pom 引入jeasy2.3.2.jar4.maven pom 配置打包5.cmd执行&#xff08;手动指定 main主程序入口&#xff09;6.EXE4J打包jar包&#xff0c;生成exe可执行文件 1.JEasyOpc下载 jeasyopc源码下载&…

5 WPF中的Page页面的使用

以下是一个简单的WPF示例&#xff0c;演示如何在三个Page之间进行导航切换&#xff0c;使用Frame控件作为导航容器&#xff0c;并包含基本的导航按钮&#xff08;前进/后退/主页&#xff09; Page类更简单&#xff0c;比Window更精简。 代码见下文以及资源文件&#xff1a; htt…

基于51单片机的音乐盒点阵屏proteus仿真

地址&#xff1a; https://pan.baidu.com/s/1hYzg2icjHV8jWJdltJkKxw 提取码&#xff1a;1234 仿真图&#xff1a; 芯片/模块的特点&#xff1a; AT89C52/AT89C51简介&#xff1a; AT89C51 是一款常用的 8 位单片机&#xff0c;由 Atmel 公司&#xff08;现已被 Microchip 收…

图论:floyed算法

Floyd 算法是一种用于寻找加权图中所有顶点对之间最短路径的经典算法&#xff0c;它能够处理负权边&#xff0c;但不能处理负权环。即如果边权有负数&#xff0c;切负权边与其他边构成了环就不能用该算法。该算法的时间复杂度为 \(O(V^3)\)&#xff0c;其中 V 是图中顶点的数量…

STM32之看门狗(IWDG)

一、看门狗外设的原理与应用 背景说明 随着单片机的发展&#xff0c;单片机在家用电器、工业自动化、生产过程控制、智能仪器仪表等领域的应用越来越广泛。然而处于同一电力系统中的各种电气设备通过电或磁的联系彼此紧密相连&#xff0c;相互影响&#xff0c;由于运行方式的…

#RabbitMQ# 消息队列进阶

目录 消息可靠性 一 生产者的可靠性 1 生产者的重连 2 生产者的确认 (1 Confirm* (2 Return 二 MQ的可靠性 1 数据持久化 2 Lazy Queue* 三 消费者的可靠性 1 消费者确认机制 2 消费失败处理 3 业务幂等性 四 延迟消息 消息可靠性 在消息队列中&#xff0c;可靠性…

《计算机组成原理》第 10 章 - 控制单元的设计

目录 10.1 组合逻辑设计 10.1.1 组合逻辑控制单元框图 10.1.2 微操作的节拍安排 10.1.3 组合逻辑设计步骤 10.2 微程序设计 10.2.1 微程序设计思想的产生 10.2.2 微程序控制单元框图及工作原理 10.2.3 微指令的编码方式 1. 直接编码&#xff08;水平型&#xff09; 2.…

AstroNex空间任务智能控制研究与训练数据集

数据集概述 AstroNex空间任务智能控制研究与训练数据集是朗迪锋科技基于Multiverse平台精心打造的首个全面覆盖航天器智能控制全周期的综合数据集产品。该数据集汇集了轨道动力学、姿态控制、机器视觉、环境感知等多维度数据&#xff0c;为航天器智能算法研发提供丰富的训练与…

​​3D 几何建模工具库​Open CASCADE(OCCT)简单介绍。

​​Open CASCADE&#xff08;OCCT&#xff09;​​ 的新手&#xff0c;我会用最简单的方式帮你理解它是什么、能做什么&#xff0c;以及如何快速上手。 ​​1. OCCT 是什么&#xff1f;​​ ​​一句话定义​​&#xff1a;OCCT 是一个开源的 ​​3D 几何建模工具库​​&…

[7-1] ADC模数转换器 江协科技学习笔记(14个知识点)

1 2 3 4 5 6 7 8 9 10 11 12 13 14 DMA&#xff08;Direct Memory Access&#xff0c;直接内存访问&#xff09;是一种硬件特性&#xff0c;它允许某些硬件子系统直接访问系统的内存&#xff0c;而无需CPU的介入。这样&#xff0c;CPU就可以处理其他任务&#xff0c;从而提高系…

篇章三 基础——不可变类

目录 1.是什么 2.为什么 3.怎么做 4.构造详细的不可变类示例: 5.补充 5.1 Java标准库中的不可变类 5.2 构造不可变类进阶 1.对象包含嵌套的引用类型字段 2. 大型对象采用不可变类时,需考虑性能影响。 2.1 内存占用问题 2.2 垃圾回收压力 2.3 复制开销 2.4 优化策…

cuda ncu section 含义解释

NVIDIA Nsight Compute (NCU) 是用于分析 CUDA 程序性能的工具&#xff0c;通过 Sections 组织性能指标。用户提供的 24 个 Sections 涵盖了计算、内存、调度、互连和可视化等方面。本报告详细解释每个 Section 的含义、用途及相关分析场景。 Sections 详细解析 C2CLink 含义&…

NGINX HTTP/2 全面指南开启、调优与实战

一、为什么要用 HTTP/2&#xff1f; 多路复用&#xff08;Multiplexing&#xff09; 单连接上可并发交错发送多路请求&#xff0c;避免了 HTTP/1.x 中的队头阻塞&#xff08;Head-Of-Line Blocking&#xff09;。头部压缩&#xff08;HPACK&#xff09; 对 HTTP 头部字段进行高…

手写简单的tomcat

首先&#xff0c;Tomcat是一个软件&#xff0c;所有的项目都能在Tomcat上加载运行&#xff0c;Tomcat最核心的就是Servlet集合&#xff0c;本身就是HashMap。Tomcat需要支持Servlet&#xff0c;所以有servlet底层的资源&#xff1a;HttpServlet抽象类、HttpRequest和HttpRespon…

智能体赋能效率,企业知识库沉淀价值:UMI企业智脑的双轮驱动!

智能体企业知识库&#xff1a;UMI企业智脑的核心功能与价值 在人工智能技术飞速发展的今天&#xff0c;企业智能化转型已经成为不可逆转的趋势。作为企业级AI智能体开发平台的佼佼者&#xff0c;优秘智能推出的UMI企业智脑&#xff0c;以其强大的智能体开发能力和全面的企业知…

与 PyCharm 官方沟通解决开发环境问题记录(进展:官方已推出2个新的修复版本)

​​​​​​主题&#xff1a;有关 PyCharm 中终端和环境激活问题的反馈&#xff1a;PY-81233 前言 目前进展&#xff1a; 官方已有2个修复版本推出测试。 更新方法&#xff1a; 使用JetBrains Toolbox App&#xff0c;如下图所示&#xff0c;从“其他版本”进入查看更新。…

LINUX安装运行jeelowcode后端项目(命令行)

环境准备 运行环境&#xff1a;JDK1.8开发工具&#xff1a; Idea、Maven默认已启动中间件&#xff1a;&#xff08;推荐使用宝塔&#xff09;Mysql8.0、Redis、Minio第一步&#xff1a;下载JeelowCode项目并导入IDEA中 第二步&#xff1a;导入数据库文件到mysql中&#xff0c;…