Springboot @Cacheable 配置缓存过期时间,@Cacheable配置自定义过期时间,序列化异常问题解决_HoneyZe的博客-CSDN博客

@Cacheable配置全局键过期时间

案发背景

项目使用@Cacheable注解来实现方法级别的缓存,需求中有些方法适合使用仅缓存一两小时即可,但现有的@Cacheable注解中没有直接设置缓存时间的字段,所以需要单独配置

  • @Cacheable 注解并没有给可提供键过期时间的操作, 只可使用CacheManger来管理有过期值的键
  • 需单独写一个配置文件,来管理需要缓存有过期时间的键
  • 配置文件中可配置键,值的序列化方式默认过期时间(默认不会过期)
  • SpringBoot @Cacheable注解可以基于JVM内存(每次重启应用缓存失效)
    或基于redis(缓存在redis里),本文实现为redis
  • 起初本想直接百度一个配置,怎料出现了键值序列化等各种问题,无奈只好去一点点看源码分析,此处记录了分析过程,望抛砖引玉
  • 如想直接看结果代码拉到最下面即可

问题排查分析

在使用网上百度的配置时出现了序列化异常问题,但不配置CacheConfig却能正常缓存,启用自己的配置文件却出现异常,这里说明自己的配置CacheConfig与SpringBoot的默认配置有所出入,但自己写的配置无非就是更改某些配置参数,所以解决思路就是使用SpringBoot默认的配置,来排查原因

查看SpringBoot默认配置

/**
 * @Author: ZeRen.
 */
@Configuration
public class TestConfig {

    @Autowired
    CacheManager cacheManager;

    @PostConstruct
    public void viewDefaultCacheManager() {
       RedisCacheManager redisCacheManager = (RedisCacheManager) this.cacheManager;
       System.out.println(redisCacheManager);
    }
}

IDEA点击此处即可看到SpringBoot的默认缓存配置CacheManager的默认声明所在位置IDEA中点击此处可跟踪查看配置此Bean的地方
这里就是SpringBoot的默认键值序列化配置项了
此处正是SpringBoot的缓存默认配置键值序列化配置
想看源码的同学可以按照我上面描述的使用IDEA进入源码中研究,此处就不一一描述了,这里需要注意的是这两个键值序列化方式,和一个RedisCacheConfiguration类的工厂构造方法入参的一个类加载器ClassLoader(实现为ResourceLoader)

解决代码

package com.sun.config;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cache.CacheManager;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.ResourceLoader;
import org.springframework.data.redis.cache.RedisCacheConfiguration;
import org.springframework.data.redis.cache.RedisCacheManager;
import org.springframework.data.redis.connection.RedisConnectionFactory;

import javax.annotation.PostConstruct;
import java.time.Duration;
import java.util.HashMap;
import java.util.Map;

/**
 * @Author: ZeRen.
 * @title 该配置文件大部分使用SpringBoot默认配置,仅加入了有期限缓存的键
 */
@Configuration
public class CacheConfig {

    @Autowired
    ResourceLoader resourceLoader;

    @Bean
    public CacheManager cacheManager(RedisConnectionFactory redisConnectionFactory) {
        return RedisCacheManager.builder(redisConnectionFactory)
                .cacheDefaults(getDefaultCacheConfiguration())//默认的缓存配置(没有配置键的key均使用此配置)
                .withInitialCacheConfigurations(getCacheConfigurations())
                .transactionAware() //在spring事务正常提交时才缓存数据
                .build();
    }

    private Map<String, RedisCacheConfiguration> getCacheConfigurations() {
        Map<String, RedisCacheConfiguration> configurationMap = new HashMap<>();

        //缓存键,且30秒后过期,30秒后再次调用方法时需要重新缓存
        configurationMap.put("expireKey", this.getDefaultCacheConfiguration(Duration.ofSeconds(30)));

        return configurationMap;
    }

 
    /**
     * 获取redis的缓存配置(针对于键)
     *
     * @param ttl 键过期时间
     * @return
     */
    private RedisCacheConfiguration getDefaultCacheConfiguration(Duration ttl) {
        // 获取Redis缓存配置,此处获取的为默认配置
        final RedisCacheConfiguration defaultCacheConfiguration = getDefaultCacheConfiguration();
        // 设置键过期的时间,用 java.time 下的Duration表示持续时间,进入entryTtl()方法的源码中可看到
        // 当设置为 0 即 Duration.ZERO 时表示键无过期时间,其也是默认配置
        return defaultCacheConfiguration.entryTtl(ttl);
    }

    /**
     * 获取Redis缓存配置,此处获取的为默认配置
     * 如对键值序列化方式,是否缓存null值,是否使用前缀等有特殊要求
     * 可另行调用 RedisCacheConfiguration 的构造方法
     *
     * @return
     */
    private RedisCacheConfiguration getDefaultCacheConfiguration() {
        // 注意此构造函数为 spring-data-redis-2.1.9 及以上拥有,经试验 已知spring-data-redis-2.0.9及以下版本没有此构造函数
        // 但观察源码可得核心不过是在值序列化器(valueSerializationPair)的构造中注入 ClassLoader 即可
        return RedisCacheConfiguration.defaultCacheConfig(resourceLoader.getClassLoader());
    }

}

在上面关键字段配置处都有注释描述需要注意的是,如果报错,一般是因为spring-redis-data版本不一致了,自行升级版本,或修改入参构造方法即可

流程

  • 分析问题,序列化异常推断类加载器不正确,或序列化方式不一致导致
  • 查看CacheManager的默认SpringBoot 注入的Bean的配置方式

    • ide 点击 下面@Autowired 注入的CacheManager 左侧的绿色圆圈里有箭头的按钮,进入SpringBoot 配置Bean的默认方法
    • ps:此处的CacheManager正是SpringBoot自动配置 auto-config,约定大于配置的体现
    • 观察得知,须有一个 ResourceLoader 传入,在 RedisCacheConfiguration 中的 valueSerializationPair ,即序列化键的value时使用此ClassLoader
  • 结和现有配置及Spring的默认配置,及文档注释,以上问题得以解决

依赖版本

<!-- 使用SpringBoot版本为2.1.11 -->
<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>2.1.11.RELEASE</version>
    <relativePath/>
</parent>
<!-- Redis依赖 -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

总结

本次经历虽耗费半天时间,但却体现了个SpringBoot配置问题的解决思路,仅此记录,
望抛砖引玉

Original url: Access
Created at: 2020-09-22 17:39:23
Category: default
Tags: none

请先后发表评论
  • 最新评论
  • 总共0条评论