济南市莱芜区网站信宜网站开发公司
2026/4/15 3:55:48 网站建设 项目流程
济南市莱芜区网站,信宜网站开发公司,按颜色分类的网页设计欣赏网站,网络营销优化一、 核心定义与作用 Spring Cache 不是一个具体的缓存实现#xff08;它不是 Redis#xff0c;也不是 EhCache#xff09;#xff0c;而是一套 缓存抽象层#xff08;Cache Abstraction#xff09;。 1. 核心定位 它类似于 JDBC 之于数据库。 JDBC 定义了标准接口#…一、 核心定义与作用Spring Cache不是一个具体的缓存实现它不是 Redis也不是 EhCache而是一套缓存抽象层Cache Abstraction。1. 核心定位它类似于 JDBC 之于数据库。JDBC定义了标准接口你可以底层换 MySQL 或 Oracle。Spring Cache定义了标准注解 (Cacheable)你可以底层换 Redis、Caffeine 或 ConcurrentMap而业务代码不需要修改一行。2. 核心价值代码解耦将业务逻辑Service与缓存技术细节分离。消除样板代码通过 AOP面向切面编程管理缓存不再需要手动编写redisTemplate.get/set。统一标准无论底层用什么上层注解用法一致。二、 核心注解体系Spring Cache 主要通过以下 5 个注解控制缓存行为注解核心作用执行逻辑 (通俗版)适用场景EnableCaching总开关开启 Spring 的缓存代理功能。启动类 / 配置类Cacheable查/存1. 先查缓存有则直接返回。2. 无则执行方法并将结果存入缓存。查询(Get)CachePut改/存始终执行方法并将返回值强制更新到缓存中。新增/修改(Save/Update)CacheEvict删执行方法前或后从缓存中删除指定数据。删除(Delete)Caching组合在一个方法上叠加多个操作如删 A 缓存同时删 B 缓存。复杂联动场景通用关键参数所有注解除开关外都支持以下参数value/cacheNames缓存名称对应 Redis 的 Key 前缀。key缓存 Key支持SpEL 表达式如#id,#user.name。condition事前判断。满足条件才处理缓存例如#id 10。unless事后判断。满足条件不缓存例如#result null。三、 实践案例详解以下模拟一个用户系统展示如何结合 Redis 使用。1. 查询缓存Cacheable需求查询用户缓存有直接返回无则查库并回填。同时防止缓存null值。ServicepublicclassUserService{AutowiredprivateUserMapperuserMapper;/** * value users: 对应 Redis 前缀 users:: (或自定义的 users:) * key #id: 取参数 id 作为后缀 * unless #result null: 如果查不到数据不要把 null 存进 Redis */Cacheable(valueusers,key#id,unless#result null)publicUsergetUserById(Longid){System.out.println(--- 走数据库查询 ---);returnuserMapper.selectById(id);}}2. 保持一致性CacheEvict 推荐需求修改用户信息后清理缓存保证下次查询获取最新数据。最佳实践优先使用删除模式 (Evict)而非更新模式 (Put)避免并发写导致脏数据。/** * 更新完成后删除对应的缓存 Key * key 必须与查询时的 key 生成规则一致 */CacheEvict(valueusers,key#user.id)publicvoidupdateUser(Useruser){userMapper.updateById(user);}/** * 场景删除用户或者清空整个缓存桶 * allEntries true: 删除 users 下的所有 key */CacheEvict(valueusers,allEntriestrue)publicvoidclearAllCache(){System.out.println(清空用户缓存);}3. 复杂 Key 生成 (SpEL)需求根据多个参数组合生成 Key。/** * 假设 typeVIP, page1 * Redis Key: user_list:VIP_1 */Cacheable(valueuser_list,key#type _ #page)publicListUsergetUsersByType(Stringtype,intpage){returnuserMapper.selectByType(type,page);}四、 避坑指南内部调用失效 (Self-Invocation)这是 Spring Cache 最经典的“大坑”。1. 现象描述在同一个 Service 类内部方法 A 调用带Cacheable的方法 B方法 B 的缓存注解失效每次都会执行 SQL。2. 原理图解Spring Cache 基于代理模式 (Proxy Pattern)。外部调用Controller - Proxy对象 - 拦截处理缓存 - 目标对象。内部调用this.method()是目标对象自己调用自己绕过了 Proxy 对象所以 AOP 切面无法执行。3. 代码示例与修复❌ 错误写法 (失效)ServicepublicclassUserService{// 方法 A批量查询publicListUsergetUsersByIds(ListLongids){ListUserusersnewArrayList();for(Longid:ids){// 【失效原因】这里等同于 this.getUserById(id)// 直接调用了类内部的方法没有经过 Spring 的代理类users.add(getUserById(id));}returnusers;}// 方法 B单查注解在此Cacheable(valueusers,key#id)publicUsergetUserById(Longid){System.out.println(查询数据库...);returnnewUser(id,name);}}✅ 修复方案 1自我注入 (简单有效)通过注入自身代理对象来调用强行经过代理层。ServicepublicclassUserService{// 1. 注入自己 (加上 Lazy 防止循环依赖报错)AutowiredLazyprivateUserServiceself;publicListUsergetUsersByIds(ListLongids){ListUserusersnewArrayList();for(Longid:ids){// 2. 【修复】使用代理对象 self 调用而不是 this// 流程self - Proxy - 检查缓存 - Targetusers.add(self.getUserById(id));}returnusers;}Cacheable(valueusers,key#id)publicUsergetUserById(Longid){// ...}}✅ 修复方案 2服务拆分 (架构更优)将缓存方法抽离到另一个 Service 中符合单一职责原则。// 新服务专门负责缓存操作ServicepublicclassUserCacheService{Cacheable(valueusers,key#id)publicUsergetUserById(Longid){...}}// 原服务注入上面的服务ServicepublicclassUserService{AutowiredprivateUserCacheServiceuserCacheService;publicListUsergetUsersByIds(ListLongids){// 外部调用缓存必然生效returnids.stream().map(userCacheService::getUserById).collect(Collectors.toList());}}

需要专业的网站建设服务?

联系我们获取免费的网站建设咨询和方案报价,让我们帮助您实现业务目标

立即咨询