【SpringCache】SpringCache详解及其使用,Redis控制失效时间
一、使用
在 Spring 中,使用缓存通常涉及以下步骤:
1、添加缓存依赖: 确保项目中添加了缓存相关的依赖。如果使用 Maven,可以在项目的 pom.xml 文件中添加 Spring Cache 的依赖。
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-cache</artifactId>
</dependency>
2、配置缓存: 在 Spring 配置文件中进行缓存的基本配置。如果是使用 Spring Boot,通常无需额外配置,因为 Spring Boot 提供了默认的缓存配置。
第1步:在启动类上加注解@EnableCaching
第2步:在具体方法上加注解【@CachePut、@CacheEvict、@Caching】或者在方法上添加 @Cacheable 注解: 在需要缓存的方法上使用 @Cacheable 注解,指定缓存的名称、键等信息。
3、触发缓存: 调用带有 @Cacheable 注解的方法时,Spring 会检查缓存中是否已经有了相应的结果。如果有,则直接返回缓存结果;如果没有,则执行方法体,计算结果并将结果缓存起来。
二、详解
1、@Cacheable
@Cacheable 是 Spring Framework 中的一个注解,用于声明某个方法的结果应该被缓存,以便在后续调用中可以直接返回缓存的结果,而不需要再次执行方法体。这个注解通常用于提高方法的执行效率,尤其是对于那些计算成本较高的方法。
基本用法
import org.springframework.cache.annotation.Cacheable;public class MyService {@Cacheable(value = "myCache", key = "#input")public String getResult(String input) {// 如果缓存中存在以input为key的结果,则直接返回缓存结果// 否则,执行方法体,并将结果缓存起来// ...}
}
在上述例子中:
@Cacheable 注解标注在 getResult 方法上,表示该方法的结果将会被缓存。
value 属性指定了缓存的名称(可以有多个缓存,每个缓存有一个名称)。
key 属性指定了缓存的键,这里使用了 SpEL(Spring Expression Language)表达式,表示使用 input 参数的值作为缓存的键。有时候会按照下方书写方式:
@Cacheable(value = "myCache", key = "#p0+','+#p1")
key = “#p0+‘,’+#p1” :指定缓存的键,使用了 Spring Expression Language(SpEL)的语法。#p0 表示方法的第一个参数,#p1 表示方法的第二个参数。在这个例子中,键由第一个参数和第二个参数以逗号连接而成。这意味着如果两次调用方法的时候,这两个参数的值相同,那么它们会共享相同的缓存结果。如果想要每次调用方法都修改缓存值,可以用其他拼接符连接,比如横杠、空格、冒号或者将方法名作为key值一部分。
key = "#p0 + '_' + #p1" // 使用下划线key = "#p0 + ',' + #p1 + ',' + #p2" // 添加额外的参数或常量key = "methodName + ',' + #p0 + ',' + #p1" //使用方法名作为一部分
注意事项:
@Cacheable 注解需要在 Spring 环境中生效,因此通常需要在 Spring 的配置文件中启用缓存功能。
方法的返回值类型应该是可序列化的,以便能够存储在缓存中。
缓存的键是根据 key 属性生成的,所以确保它能够唯一标识方法的输入参数。
@Cacheable 是 Spring Cache 抽象的一部分,而具体的缓存实现可以是基于不同的后端,比如基于内存、Redis、Ehcache 等。
我们在使用的时候常常会和@SqlQuery注解一块使用,如果一个方法同时使用了 @Cacheable 注解和 @SqlQuery 注解,那么其执行流程通常如下:
-
首次调用: 当方法被首次调用时,Spring 会先检查缓存(由 @Cacheable
注解管理)。如果缓存中存在对应的结果,则直接从缓存中取出并返回,而不会执行实际的 SQL 查询。 -
缓存未命中: 如果缓存中不存在对应的结果,那么方法体会被执行。在方法体中,可能会执行 SQL 查询(由 @SqlQuery
注解管理),并将查询结果映射为方法的返回类型。 -
结果缓存: 如果启用了 @Cacheable
注解,方法执行完毕后,计算得到的结果会被缓存起来,以便下次相同的方法调用可以直接从缓存中获取。
所以,如果缓存中已有相应的结果,实际的 SQL 查询可能不会执行。这样可以有效减轻对数据库的访问压力,提高方法的执行效率。在使用这两个注解时,确保缓存的键和 SQL 查询的参数不会引起混淆,以便正确地识别和管理缓存。
2、@CacheEvict
@CacheEvict 是 Spring 框架中用于清除缓存的注解。它用于标记一个方法,在方法执行时会清除指定缓存中的数据。以下是 @CacheEvict 注解的主要属性和用法:
主要属性:
value(或 cacheNames): 指定要清除的缓存的名称,可以是一个字符串数组。例如:@CacheEvict(value = “myCache”) 或 @CacheEvict(cacheNames = {“cache1”, “cache2”})。
key: 指定用于生成缓存键的 SpEL 表达式。例如:@CacheEvict(value = "myCache", key = "#userI
condition: 指定清除缓存的条件,是一个 SpEL 表达式,如果表达式的值为 false,则不会清除缓存。例如:@CacheEvict(value = "myCache", condition = "#userId > 0")。
allEntries: 如果设置为 true,则会清除指定缓存中的所有条目。例如:@CacheEvict(value = "myCache", allEntries = true)。
beforeInvocation: 如果设置为 true,则在方法调用之前清除缓存;如果设置为 false(默认),则在方法调用之后清除缓存。例如:@CacheEvict(value = "myCache", beforeInvocation = true)。
示例用法:
@Service
public class MyService {@CacheEvict(value = "myCache", key = "#userId")public void clearCacheByUserId(long userId) {// 此方法执行时,会清除名为 "myCache" 中 key 为 userId 的缓存条目}@CacheEvict(value = "myCache", allEntries = true)public void clearEntireCache() {// 此方法执行时,会清除名为 "myCache" 中的所有缓存条目}
}
在上述例子中,clearCacheByUserId 方法用于清除名为 “myCache” 中 key 为 userId 的缓存条目,而 clearEntireCache 方法用于清除 “myCache” 中的所有缓存条目。
请注意,@CacheEvict 注解通常用于在方法执行时清除缓存。如果方法执行抛出异常,缓存清除可能不会发生,除非设置了 beforeInvocation = true。因此,要确保清除缓存的操作是安全的,不会因为异常而导致缓存清除失败。
3、@CachePut
@CachePut 是 Spring 框架中用于更新缓存的注解。它用于标记一个方法,在方法执行时会将结果放入缓存中。与 @Cacheable 不同,@CachePut 不会先检查缓存中是否已有结果,而是直接将方法的返回值放入缓存中。
以下是 @CachePut 注解的主要属性和用法:
主要属性:
value(或 cacheNames): 指定要更新的缓存的名称,可以是一个字符串数组。例如:@CachePut(value = "myCache") 或 @CachePut(cacheNames = {"cache1", "cache2"})。
key: 指定用于生成缓存键的 SpEL 表达式。例如:@CachePut(value = "myCache", key = "#userId")。
condition: 指定放入缓存的条件,是一个 SpEL 表达式,如果表达式的值为 false,则不会放入缓存。例如:@CachePut(value = "myCache", condition = "#userId > 0")。
unless: 与 condition 相反,如果表达式的值为 true,则不会放入缓存。例如:@CachePut(value = "myCache", unless = "#result == null")。
示例用法:
@Service
public class MyService {@CachePut(value = "myCache", key = "#userId")public String updateCacheByUserId(long userId) {// 此方法执行时,会将返回值放入名为 "myCache" 中 key 为 userId 的缓存条目// 注意:不会先检查缓存中是否已有结果,直接将方法的返回值放入缓存中// ...return "Updated Value";}
}
在上述例子中,updateCacheByUserId 方法用于将返回值放入名为 “myCache” 中 key 为 userId 的缓存条目。这个方法在执行时,不会先检查缓存中是否已有结果,而是直接将方法的返回值放入缓存中。
@CachePut 通常用于在更新操作之后,将最新的结果放入缓存,以保持缓存的一致性。需要注意的是,与 @Cacheable 不同,@CachePut 不会阻止方法的执行,即使缓存操作失败也不会影响方法的正常执行。
4、@Caching
@Caching 是 Spring 框架中用于组合多个缓存注解的注解。它允许同时在一个方法上使用多个缓存相关的注解,包括 @Cacheable、@CachePut、@CacheEvict 等,从而提供更灵活的缓存配置。
示例用法:
@Service
public class MyService {@Caching(cacheable = {@Cacheable(value = "cache1", key = "#userId")},put = {@CachePut(value = "cache2", key = "#result.id")})public User getUserById(long userId) {// 先尝试从 "cache1" 缓存中获取结果// 如果获取成功,则返回结果,不执行方法体// 如果获取失败,则执行方法体,并将结果放入 "cache1" 和 "cache2" 缓存中// ...return new User(userId, "John Doe");}
}
在上述例子中,@Caching 注解用于同时使用 @Cacheable 和 @CachePut 注解。具体来说:
@Cacheable 注解用于尝试从名为 “cache1” 的缓存中获取结果,如果获取成功,则直接返回结果,不执行方法体。
@CachePut 注解用于将方法的返回值放入名为 “cache2” 的缓存中,不论缓存中是否已有相同的键。
通过 @Caching 注解,可以更灵活地组合多个缓存注解,以满足复杂的缓存需求。
三、cache失效时间配置
1、注解属性中设置
在使用 @Cacheable 注解时,你可以通过设置 expire 或 expireAfterWrite 属性来配置缓存的失效时间。这取决于使用的缓存管理器,因为不同的缓存管理器可能支持不同的配置。
例如,如果你使用的是 Spring Boot,并且底层的缓存管理器是基于 Caffeine 的,你可以使用 expireAfterWrite 属性来设置失效时间。以下是一个示例:
@Cacheable(value = "myCache", key = "#input", expireAfterWrite = 5, timeUnit = TimeUnit.MINUTES)
public String getResult(String input) {// 如果缓存中存在以 input 为 key 的结果,则直接返回缓存结果// 否则,执行方法体,并将结果缓存起来,缓存时间为 5 分钟// ...
}
在上述示例中:
expireAfterWrite = 5 表示缓存项在写入后经过 5 分钟过期。
timeUnit = TimeUnit.MINUTES 表示时间单位是分钟。
具体的配置方式取决于使用的缓存管理器。如果你使用的是其他缓存管理器,例如 EhCache、Redis 等,具体的配置方式可能会有所不同。在配置文件或注解中查看相关属性,以确保正确地设置缓存的失效时间。
2、redis管理器设置
当使用 Redis 作为缓存管理器时,可以通过 Spring Boot 的配置文件或 Java 配置来设置 Redis 缓存的失效时间。以下是一个示例:
-
在 Spring Boot 的配置文件中设置 Redis 缓存失效时间: 在 application.properties 或
application.yml 文件中添加以下配置:# Redis 服务器地址 spring.redis.host=localhost # Redis 服务器端口 spring.redis.port=6379 # 缓存的默认失效时间,单位秒 spring.cache.redis.time-to-live=600 -
上述配置中的 spring.cache.redis.time-to-live
表示缓存的默认失效时间为600秒(10分钟)。这个值会应用于所有通过 @Cacheable、@CachePut 注解配置的缓存。在 Java 配置中设置 Redis 缓存失效时间: 如果你使用 Java 配置类,可以在配置类中通过
RedisCacheConfiguration 来设置缓存的失效时间。以下是一个示例:java @Configuration @EnableCaching public class CacheConfigextends CachingConfigurerSupport {@Beanpublic RedisCacheManager cacheManager(RedisConnectionFactory connectionFactory) {RedisCacheConfiguration config = RedisCacheConfiguration.defaultCacheConfig().entryTtl(Duration.ofSeconds(600)); // 设置缓存失效时间为600秒(10分钟)return RedisCacheManager.builder(connectionFactory).cacheDefaults(config).build();} }在上述配置中,
entryTtl(Duration.ofSeconds(600))表示设置缓存的失效时间为600秒。这个值同样会应用于所有通过 @Cacheable、@CachePut 注解配置的缓存。
根据具体的需求,你可以根据缓存的特定性配置不同的失效时间。上述示例中仅作为演示,实际应用中可以根据业务需求来设置合适的缓存失效时间。
相关文章:
【SpringCache】SpringCache详解及其使用,Redis控制失效时间
一、使用 在 Spring 中,使用缓存通常涉及以下步骤: 1、添加缓存依赖: 确保项目中添加了缓存相关的依赖。如果使用 Maven,可以在项目的 pom.xml 文件中添加 Spring Cache 的依赖。 <dependency><groupId>org.spring…...
MyBatis的基本使用及常见问题
MyBatis 前言MyBatis简介MyBatis快速上手Mapper代理开发增删改查环境准备配置文件完成增删改查查询添加修改删除 参数传递注解完成增删改查 前言 JavaWeb JavaWeb是用Java技术来解决相关Web互联网领域的技术栈。 MySQL数据库与SQL语言 MySQL:开源的中小型数据库。…...
[RoarCTF2019] TankGame
不多说,用dnspy反编译data文件夹中的Assembly-CSharp文件 使用分析器分析一下可疑的FlagText 发现其在WinGame中被调用,跟进WinGame函数 public static void WinGame(){if (!MapManager.winGame && (MapManager.nDestroyNum 4 || MapManager.n…...
相比于其他流处理技术,Flink的优点在哪?
Apache Flink 是一个开源的流处理框架,用于在高吞吐量和低延迟的情况下进行大规模数据流的处理。Flink 以其在流处理领域的性能而闻名,相比于其他流处理技术,Flink 提供了一些独特的特性和优化,使其在某些情况下更快。以下是 Flin…...
react中使用ref属性获取元素,并判断该元素内是否含有子元素
在react中,可以使用ref属性来获取到一个元素的引用,然后再使用ref.current来访问该元素的DOM节点,使用DOM API来判断这个元素是否含有子元素,要判断一个元素是否含有子元素,可以使用hasChildNodes(),其返回…...
idea 如何快速拉取新分支
方式1 (快捷键:CtrlShift~) 方式2:(快捷键:Alt9)...
【经验分享】日常开发中的故障排查经验分享(一)
目录 简介CPU飙高问题1、使用JVM命令排查CPU飙升100%问题2、使用Arthas的方式定位CPU飙升问题3、Java项目导致CPU飙升的原因有哪些?如何解决? OOM问题(内存溢出)1、如何定位OOM问题?2、OOM问题产生原因 死锁问题的定位…...
关于Unity使用图片字体示例
1.使用TexturePacker打包图集 下载地址 TexturePacker - Create Sprite Sheets for your game! 2.准备好数字图 3. 导入图片 4. 打包图集需要的设置 将重心点设置为左下方 点击回车 > 后点击回 >到精灵列表 选择导出的格式 导出后的内容 >导入unity 导入 >…...
开源大语言模型简记
文章目录 开源大模型LlamaChinese-LLaMA-AlpacaLlama2-ChineseLinlyYaYiChatGLMtransformersGPT-3(未完全开源)BERTT5QwenBELLEMossBaichuan其他...
python高级代码
目录 列表推导式和生成器表达式:使用简洁的语法来生成列表和生成器。 装饰器:用于修改函数行为的函数。 上下文管理器:用于管理资源的对象,可以使用with语句来自动管理资源的分配和释放。 多线程和多进程编程:使用…...
透彻掌握GIT基础使用
网址 https://learngitbranching.js.org/?localezh_CN 清屏 clear重新开始reset...
二、类与对象(三)
17 初始化列表 17.1 初始化列表的引入 之前我们给成员进行初始化时,采用的是下面的这种方式: class Date { public:Date(int year, int month, int day)//构造函数{_year year;_month month;_day day;} private:int _year;int _month;int _day; };…...
CentOS 7 Tomcat服务的安装
前提 安装java https://blog.csdn.net/qq_36940806/article/details/134945175?spm1001.2014.3001.5501 1. 下载 wget https://mirrors.tuna.tsinghua.edu.cn/apache/tomcat/tomcat-9/v9.0.84/bin/apache-tomcat-9.0.84.tar.gzps: 可选择自己需要的版本下载安装https://mir…...
文件夹共享功能的配置 以及Windows server2012防火墙的配置
目录 一. 配置文件夹共享功能 1.1 为什么需要配置文件夹共享功能 1.2 配置文件夹共享 1.3 访问共享文件夹 1.4 配置取消 用户名和密码认证 二. windows server 2012防火墙配置 思维导图 一. 配置文件夹共享功能 1.1 为什么需要配置文件夹共享功能 我们在工作和生活中经…...
前端使用高德api的AMap.Autocomplete无效,使用AMap.Autocomplete报错
今天需要一个坐标拾取器,需要一个输入框输入模糊地址能筛选的功能 查看官方文档,有一个api可以直接满足我们的需求 AMap.Autocomplete 上代码 AMapLoader.load({"key": "你的key", // 申请好的Web端开发者Key,首次调…...
反转链表、链表的中间结点、合并两个有序链表(leetcode 一题多解)
一、反转链表 给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。 力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台 思路一:翻转单链表指针方向 这里解释一下三个指针的作用: n1࿱…...
深度学习中的Dropout
1 Dropout概述 1.1 什么是Dropout 在2012年,Hinton在其论文《Improving neural networks by preventing co-adaptation of feature detectors》中提出Dropout。当一个复杂的前馈神经网络被训练在小的数据集时,容易造成过拟合。为了防止过拟合ÿ…...
MySQL 中的 ibdata1 文件过大如何处理?
ibdata1 是什么文件? ibdata1 是InnoDB的共有表空间,默认情况下会把表空间存放在一个名叫 ibdata1的文件中,日积月累会使该文件越来越大。 ibdata1 文件过大的解决办法 使用独享表空间,将表空间分别单独存放。MySQL开启独享表空…...
Weblogic反序列化远程命令执行(CVE-2019-2725)
漏洞描述: CVE-2019-2725是一个Oracle weblogic反序列化远程命令执行漏洞,这个漏洞依旧是根据weblogic的xmldecoder反序列化漏洞,通过针对Oracle官网历年来的补丁构造payload来绕过。 复现过程: 1.访问ip:port 2.可…...
鸿蒙组件数据传递:ui传递、@prop、@link
鸿蒙组件数据传递方式有很多种,下面详细罗列一下: 注意: 文章内名词解释: 正向:父变子也变 逆向:子变父也变 **第一种:直接传递 - 特点:1、任何数据类型都可以传递 2、不能响应式…...
XCTF-web-easyupload
试了试php,php7,pht,phtml等,都没有用 尝试.user.ini 抓包修改将.user.ini修改为jpg图片 在上传一个123.jpg 用蚁剑连接,得到flag...
8k长序列建模,蛋白质语言模型Prot42仅利用目标蛋白序列即可生成高亲和力结合剂
蛋白质结合剂(如抗体、抑制肽)在疾病诊断、成像分析及靶向药物递送等关键场景中发挥着不可替代的作用。传统上,高特异性蛋白质结合剂的开发高度依赖噬菌体展示、定向进化等实验技术,但这类方法普遍面临资源消耗巨大、研发周期冗长…...
iPhone密码忘记了办?iPhoneUnlocker,iPhone解锁工具Aiseesoft iPhone Unlocker 高级注册版分享
平时用 iPhone 的时候,难免会碰到解锁的麻烦事。比如密码忘了、人脸识别 / 指纹识别突然不灵,或者买了二手 iPhone 却被原来的 iCloud 账号锁住,这时候就需要靠谱的解锁工具来帮忙了。Aiseesoft iPhone Unlocker 就是专门解决这些问题的软件&…...
CMake 从 GitHub 下载第三方库并使用
有时我们希望直接使用 GitHub 上的开源库,而不想手动下载、编译和安装。 可以利用 CMake 提供的 FetchContent 模块来实现自动下载、构建和链接第三方库。 FetchContent 命令官方文档✅ 示例代码 我们将以 fmt 这个流行的格式化库为例,演示如何: 使用 FetchContent 从 GitH…...
【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)
骨骼动画基础 骨骼动画是 3D 计算机图形中常用的技术,它通过以下两个主要组件实现角色动画。 骨骼系统 (Skeleton):由层级结构的骨头组成,类似于人体骨骼蒙皮 (Mesh Skinning):将模型网格顶点绑定到骨骼上,使骨骼移动…...
【碎碎念】宝可梦 Mesh GO : 基于MESH网络的口袋妖怪 宝可梦GO游戏自组网系统
目录 游戏说明《宝可梦 Mesh GO》 —— 局域宝可梦探索Pokmon GO 类游戏核心理念应用场景Mesh 特性 宝可梦玩法融合设计游戏构想要素1. 地图探索(基于物理空间 广播范围)2. 野生宝可梦生成与广播3. 对战系统4. 道具与通信5. 延伸玩法 安全性设计 技术选…...
Python Ovito统计金刚石结构数量
大家好,我是小马老师。 本文介绍python ovito方法统计金刚石结构的方法。 Ovito Identify diamond structure命令可以识别和统计金刚石结构,但是无法直接输出结构的变化情况。 本文使用python调用ovito包的方法,可以持续统计各步的金刚石结构,具体代码如下: from ovito…...
C#中的CLR属性、依赖属性与附加属性
CLR属性的主要特征 封装性: 隐藏字段的实现细节 提供对字段的受控访问 访问控制: 可单独设置get/set访问器的可见性 可创建只读或只写属性 计算属性: 可以在getter中执行计算逻辑 不需要直接对应一个字段 验证逻辑: 可以…...
在 Spring Boot 中使用 JSP
jsp? 好多年没用了。重新整一下 还费了点时间,记录一下。 项目结构: pom: <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://ww…...
C++实现分布式网络通信框架RPC(2)——rpc发布端
有了上篇文章的项目的基本知识的了解,现在我们就开始构建项目。 目录 一、构建工程目录 二、本地服务发布成RPC服务 2.1理解RPC发布 2.2实现 三、Mprpc框架的基础类设计 3.1框架的初始化类 MprpcApplication 代码实现 3.2读取配置文件类 MprpcConfig 代码实现…...
