redis@Cacheable注解unless用法

本文介绍如何使用 Spring 的 @Cacheable 注解中的 unless 属性来避免将空结果或特定返回值缓存到 Redis 中。通过具体示例代码解释不同返回类型(如 boolean、string 和 list)的缓存条件配置。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在这里插入图片描述

unless作用 : 不缓存满足条件的数据,业务场景不细说了,直接看代码吧

unless = "#result == null"

#result就是你的返回函数,现在很多返回函数都是封装一层来作为统一出口,先看看我的统一出口

@Cacheable(value = CacheConstant.SUBJECT_PRACTICE_OPTIONS_TYPE_CACHE,
            key="'subject_practice_options_'+#subjectPractice.getId()",
            unless = "#result?.result.size() == 0")
    public Result getCorrectAnswer(Bean bean){
        try{
            if(null == bean.getId()){
                return Result.error("脑袋瓜子嗡嗡的~~~");
            }
            List<Bean> beanList = BeanService.get(bean.getId());
            return Result.OK(list);
        }catch(Exception e){
            e.printStackTrace();
            return Result.error("脑袋瓜子嗡嗡的~~~");
        }
    }

在这里插入图片描述
unless = “#result?.result.size() == 0” 这一段的意思就是说如果查出来的list的size等于0就不缓存到redis中
有人会说为什么下面这种写法不行

@Cacheable(cacheNames="book", unless="#result?.size() > 0")
public List<Book> findBook(String name)

注意2段代码的返回值,前面是我封装过的统一出口,数据库取到的list被我放到了返回值函数中的result中所以要result?.result先获取到list在进行逻辑判断

记住 : 第一个#result代表的是你这个方法的返回值

封装的都会操作了单独返回的boolean,string,list就更简单了
返回boolean类型 : unless = “!#result” 当返回false,才进行缓存
返回string类型 : unless = “#result eq ‘123’” 返回值是123的不缓存
返回string类型 : unless = “#result == null” 返回值是null的不缓存
返回list类型 : 上面有了自己看吧

### 关于 `@Cacheable` 注解中设置过期时间常见问题与解决方法 #### 设置全局和局部过期时间冲突 当在应用程序中既设置了全局缓存过期时间又针对特定缓存名称设置了不同过期时间时,可能会遇到冲突情况。通常情况下,更具体的配置会覆盖较通用的设定。例如,在 YAML 文件里指定全局默认超时时,可以同时为某些特别命名的空间提供个性化的 TTL (Time To Live),这样就不会影响到其他未指明TTL值的数据项[^4]。 ```yaml spring: cache: cache-names: caches redis: time-to-live: 10m # 全局缓存过期时间为10分钟 time-to-live: "cache1": 5m # 特定缓存'cache1'的过期时间为5分钟 ``` #### 缓存更新后旧数据仍然存在一段时间 有时即使已经成功刷新了某个被标记有 `@Cacheable` 方法的结果集,但由于之前版本的数据尚未达到其预设的有效期限而继续保留着。为了避免这种情况发生,可以通过调整策略使新产生的记录立即生效或将现有条目提前淘汰掉。一种做法是在每次调用服务端点前先清除关联键对应的存储位置;另一种则是利用条件参数控制是否允许读取已有副本[^3]。 ```java // 清除特定key下的所有缓存 @CacheEvict(value="users", allEntries=true) public void clearUserCaches() {} // 或者基于表达式的删除操作 @CachePut(value="items", key="#id") public Item updateItem(@PathVariable Long id, ...) { ... } ``` #### 动态计算缓存有效期 对于一些场景下可能希望依据业务逻辑动态决定每笔交易应该保存多久,则可以在使用 `@Cacheable` 注解的同时传递额外属性给它——比如通过 SpEL 表达式来获取当前上下文中可用的信息作为输入源之一参与运算得出最终结果。这使得开发者能够更加灵活地管理资源生命周期而不必受限于固定的数值范围之内[^2]。 ```java @Cacheable( value = {"products"}, unless = "#result == null", sync = true, condition = "@environment.getProperty('feature.toggle.dynamic.expiry')=='true'", keyGenerator = "customKeyGen" ) public Product getProductById(Long productId){ return productRepository.findById(productId).orElse(null); } ```
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值