优雅的缓存框架:SpringCache增强支持自定义过期时间TTL

本文介绍了如何增强SpringCache,使其支持自定义缓存过期时间(TTL)。通过定义CachceTTL注解、AOP拦截保存TTL信息,并在Redis写入时设置,实现了按需配置缓存过期时间的功能。详细步骤包括定义注解、实战缓存操作、AOP拦截和自定义RedisCacheManager的配置。

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

默认SpringCache是不支持自定义设置TTL的,redis的只支持配置一个统一的TTL,但是这可能并不符合业务的需求。

下面开始增强SpringCache让它支持配置化TTL

定义CachceTTL注解


/**
 * 缓存有效时长, 单位: 秒, 若写在类上则类中所有方法都继承此时间
 */
@Target({
   
   ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface CacheTTL {
   
   

    /**
     * 缓存有效时长, 单位: 秒
     */
    long value() default -1;
}

实战

缓存

/**
 * 生成验证码图片, 返回验证码文字, 并把验证码文字缓存
 */
@CacheTTL(120L)
@CachePut(key = "#token", unless = "#result == null")
public String generateCaptchaImage(String token, OutputStream os) throws IOException {
   
   
    String captchaText = ImageCaptchaGenerator.generateCaptchaText(candidateString, 4);
    ImageCaptchaGenerator.writeCaptchaImageToOutputStream(captchaText, width, height, os);
    return captchaText;
}

移除缓存

/**
 * 移除缓存
 */
@CacheTTL(120L)
@CacheEvict(key = "#token")
public void removeCaptchaCodeCache(String token) {
   
    }

实现原理

  1. 使用AOP拦截CacheTTL注解的方法或类里的所有方法
  2. 在写入redis之前把TTL信息保存到ThreadLocal中
  3. 在redis写入时取出ThreadLocal中的TTL并设置

Aop拦截并保存TTL

@Component
@Aspect
### Spring Cache 支持过期时间数据类型 在 Spring Cache 中,具体的支持取决于所使用的底层缓存提供者。对于 Redis 这样的缓存解决方案来说,设置缓存条目的过期时间可以通过多种方式完成。 当使用 `@Cacheable` 或其他与缓存有关的注解时,并未直接定义用于表示过期时间的数据类型的字段。相反,在配置缓存管理器(例如 `RedisCacheConfiguration` 和 `RedisCacheManager`)的过程中指定这些属性[^1]。 针对 Redis 实现而言,过期时间通常通过 Java 的 `Duration` 类来设定,该类允许以灵活的方式表达时间段,如秒数、分钟数等。下面是一个如何利用 `Duration` 来为 Redis 缓存项设置默认生存时间 (TTL) 的例子: ```java import org.springframework.context.annotation.Bean; import org.springframework.data.redis.cache.RedisCacheConfiguration; import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer; import org.springframework.data.redis.serializer.RedisSerializationContext; @Bean public RedisCacheConfiguration cacheConfig() { return RedisCacheConfiguration.defaultCacheConfig() .entryTtl(Duration.ofMinutes(60)) // 设置 TTL 为 60 分钟 .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(new GenericJackson2JsonRedisSerializer())); } ``` 此代码片段展示了怎样创建一个自定义的 `RedisCacheConfiguration` Bean 并应用了一个固定的 TTL 值给所有的缓存条目。这里采用的是 `Duration.ofMinutes()` 方法来构建持续时间为一小时的时间间隔对象;当然也可以根据需求调整单位和数值[^3]。 除了全局性的 TTL 配置外,某些情况下可能希望对特定的方法调用单独定制化其缓存的有效期限。这一般是在业务逻辑层面上处理而不是依靠于简单的注解参数传递实现。不过需要注意的是并非所有缓存库都提供了如此细粒度级别的控制选项。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值