news 2026/4/18 8:01:22

使用Spring自带的缓存注解维护数据一致性

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
使用Spring自带的缓存注解维护数据一致性

简介

Cache(缓存),已经是项目中不可缺失的存在,登录时,存储用户信息、操作权限、Token 等,高并发场景,存储热点信息、实时信息等。按照类型分类,可分为本地缓存和分布式缓存,前者重启服务失效,后者可以持久化,按照设置的过期时间或策略失效。

一旦项目中需要使用到缓存,就需要考虑到数据一致性问题,即缓存数据与数据库数据的一致性问题,本文介绍在 Spring Boot 项目中,如何使用 Spring 自带的注解来进行数据一致性的维护。

整合

Spring 自带了缓存维护的注解,如果你的项目就是 Spring/Spring Boot 项目,不需要额外引入依赖

但如果你需要缓存不会因项目重启而失效,可以引入 Redis,把 Redis 当作缓存容器,需引入 Redis 依赖。基本所有的项目都会用到 Redis,当然引入了就需要保证系统与 Redis 的连通。

<!-- 引入 Redis --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency>

在项目中创建一个 Redis 缓存配置类,里面对缓存进行统一的配置,如过期时间等。

importorg.springframework.cache.annotation.EnableCaching;importorg.springframework.context.annotation.Bean;importorg.springframework.context.annotation.Configuration;importorg.springframework.data.redis.cache.RedisCacheConfiguration;importorg.springframework.data.redis.cache.RedisCacheManager;importorg.springframework.data.redis.connection.RedisConnectionFactory;importorg.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;importorg.springframework.data.redis.serializer.RedisSerializationContext;importorg.springframework.data.redis.serializer.StringRedisSerializer;importjava.time.Duration;/** * Redis 缓存配置 */@Configuration@EnableCachingpublicclassRedisCacheConfig{@BeanpublicRedisCacheManagercacheManager(RedisConnectionFactoryconnectionFactory){RedisCacheConfigurationconfig=RedisCacheConfiguration.defaultCacheConfig().entryTtl(Duration.ofMinutes(30)).serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(newStringRedisSerializer())).serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(newGenericJackson2JsonRedisSerializer()));returnRedisCacheManager.builder(connectionFactory).cacheDefaults(config).build();}}

配置文件中,添加 Redis 配置。

spring:redis:host:localhostport:6379

使用

配合下面几个注解使用,可以加在接口上,如下,cacheNames是设置缓存名称,key设置缓存的 key 值,可以用#与方法入参关联,如#id表示 key 值取自方法入参 id 的值。

  • @Cacheable(cacheNames = “缓存名”, key = “#id”):添加缓存;

  • @CachePut(cacheNames = “缓存名”, key = “#id”):执行方法后,添加缓存;

  • @CacheEvict(cacheNames = “缓存名”, key = “#id”):使缓存失效;

需要注意的是,这套缓存作用的机制是,关联方法入参与返回,当方法入参相同时,返回缓存中的结果,不再执行代码。所以只有相同的入参才能用得上缓存。

验证

创建下面这四个接口,查询方法添加缓存,新增方法执行后添加缓存,删除和更新方法删除缓存。

importorg.example.service.CacheService;importorg.springframework.cache.annotation.CacheEvict;importorg.springframework.cache.annotation.CachePut;importorg.springframework.cache.annotation.Cacheable;importorg.springframework.web.bind.annotation.*;importjavax.annotation.Resource;@RestController@RequestMapping("/cache")publicclassCacheController{@ResourceprivateCacheServicecacheService;@GetMapping("/getOne")@Cacheable(cacheNames="cacheName",key="#id")publicStringgetOne(@RequestParamStringid){returncacheService.getOne();}@PostMapping("/createOne")@CachePut(cacheNames="cacheName",key="#id")publicStringcreateOne(@RequestParamStringid){returncacheService.createOne();}@PostMapping("/deleteOne")@CacheEvict(cacheNames="cacheName",key="#id")publicStringdeleteOne(@RequestParamStringid){returncacheService.deleteOne();}@PostMapping("/updateOne")@CacheEvict(cacheNames="cacheName",key="#id",allEntries=true)publicStringupdateOne(@RequestParamStringid){returncacheService.updateOne();}}

对应的实现类代码。

importorg.springframework.stereotype.Service;@ServicepublicclassCacheServiceImplimplementsCacheService{@OverridepublicStringgetOne(){return"Select is a cache data.";}@OverridepublicStringcreateOne(){return"Create is a cache data.";}@OverridepublicStringdeleteOne(){return"Deleted is a cache data.";}@OverridepublicStringupdateOne(){return"Updated is a cache data.";}}

启动项目,测试一下。

首次查询,断点卡在查询方法实现类这里,执行了实现类方法。

再次查询,直接返回了结果。

Redis 中增加了一个缓存。

参数换一下,传入 id=1。

发送请求,断点卡住了,说明没走缓存。

新增了一个 id=1 的缓存。

再试下更新方法,更新 id=1 的记录。

发送请求,查看 Redis,缓存都没了。

本来可以只让符合条件的,即 id=1 的缓存失效,上面都失效了,是因为我在更新方法的注解上额外加了一个属性。

allEntries = true,名称相同的所有缓存都失效,默认 false。

再试下,删除方法我没加这个属性。

调用删除方法,删除 id=1 的记录。

这回只有 id=1 的缓存被删除。

另外再试下创建方法,创建方法会再执行完创建方法后,主动将返回值添加到缓存中。

实际开发中,可以在某个创建接口完成后,将完整的对象数据返回,无缝衔接,就不用首次查询还要走一遍实现层。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/10 23:17:37

3步搞定黑苹果:OpCore Simplify让EFI配置简单到不可思议

3步搞定黑苹果&#xff1a;OpCore Simplify让EFI配置简单到不可思议 【免费下载链接】OpCore-Simplify A tool designed to simplify the creation of OpenCore EFI 项目地址: https://gitcode.com/GitHub_Trending/op/OpCore-Simplify 还在为复杂的OpenCore配置而头疼吗…

作者头像 李华
网站建设 2026/4/18 5:35:40

Windows 11系统优化终极指南:告别卡顿,重获丝滑体验

Windows 11系统优化终极指南&#xff1a;告别卡顿&#xff0c;重获丝滑体验 【免费下载链接】Win11Debloat 一个简单的PowerShell脚本&#xff0c;用于从Windows中移除预装的无用软件&#xff0c;禁用遥测&#xff0c;从Windows搜索中移除Bing&#xff0c;以及执行各种其他更改…

作者头像 李华
网站建设 2026/4/18 3:51:03

AI舞蹈教学系统:Holistic Tracking动作匹配实战案例

AI舞蹈教学系统&#xff1a;Holistic Tracking动作匹配实战案例 1. 技术背景与应用价值 随着虚拟现实、元宇宙和AI驱动内容创作的兴起&#xff0c;对高精度、低延迟的人体动作捕捉技术需求日益增长。传统动捕设备成本高昂、部署复杂&#xff0c;而基于单目摄像头的AI视觉方案…

作者头像 李华
网站建设 2026/4/18 3:50:04

BiliTools:2026年B站资源下载的革命性工具

BiliTools&#xff1a;2026年B站资源下载的革命性工具 【免费下载链接】BiliTools A cross-platform bilibili toolbox. 跨平台哔哩哔哩工具箱&#xff0c;支持视频、音乐、番剧、课程下载……持续更新 项目地址: https://gitcode.com/GitHub_Trending/bilit/BiliTools …

作者头像 李华
网站建设 2026/4/17 13:55:03

BiliTools终极指南:2026年最强B站下载神器完全解析

BiliTools终极指南&#xff1a;2026年最强B站下载神器完全解析 【免费下载链接】BiliTools A cross-platform bilibili toolbox. 跨平台哔哩哔哩工具箱&#xff0c;支持视频、音乐、番剧、课程下载……持续更新 项目地址: https://gitcode.com/GitHub_Trending/bilit/BiliToo…

作者头像 李华
网站建设 2026/4/18 3:49:05

Holistic Tracking技术解析:Google管道优化的秘密

Holistic Tracking技术解析&#xff1a;Google管道优化的秘密 1. 技术背景与核心挑战 在增强现实&#xff08;AR&#xff09;、虚拟主播&#xff08;Vtuber&#xff09;和元宇宙等前沿应用中&#xff0c;对用户全身动作的实时、高精度感知成为关键需求。传统方案通常采用多个…

作者头像 李华