Spring Boot Cache

Spring Boot Cache

一、介绍

​ spring cache 是spring3版本之后引入的一项技术,可以简化对于缓存层的操作,spring cache与springcloud stream类似,都是基于抽象层,可以任意切换其实现。

​ 其核心是CacheManagerCache这两个接口,所有由spring整合的cache都要实现这两个接口、Redis的实现类则是 RedisCacheRedisManager

二、使用

Ⅰ、查询

  1. 需要导入的依赖
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-cache</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
  1. 编写对于cache的配置
@EnableCaching
@SpringBootConfiguration
public class CacheConfig {

    @Autowired
    private RedisConnectionFactory connectionFactory;

    @Bean // 如果有多个CacheManager的话需要使用@Primary直接指定那个是默认的
    public RedisCacheManager cacheManager() {
        RedisSerializer<String> redisSerializer = new StringRedisSerializer();
        Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(Object.class);

        ObjectMapper om = new ObjectMapper();
        // 防止在序列化的过程中丢失对象的属性
        om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
        // 开启实体类和json的类型转换
        om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        jackson2JsonRedisSerializer.setObjectMapper(om);

        // 配置序列化(解决乱码的问题)
        RedisCacheConfiguration config = RedisCacheConfiguration.defaultCacheConfig()   .serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(redisSerializer))             .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer))
                // 不缓存空值
                .disableCachingNullValues()
                // 1分钟过期
                .entryTtl(Duration.ofMinutes(1))
                ;
        RedisCacheManager cacheManager = RedisCacheManager.builder(connectionFactory)
                .cacheDefaults(config)
                .build();
        return cacheManager;
    }
}

进行以上配置即可使用springboot cache了,还有一个key的生成策略的配置(可选)

@Bean
public KeyGenerator keyGenerator() {
    return (target, method, params) -> {
        StringBuffer key = new StringBuffer();
        key.append(target.getClass().getSimpleName() + "#" + method.getName() + "(");
        for (Object args : params) {
            key.append(args + ",");
        }
        key.deleteCharAt(key.length() - 1);
        key.append(")");
        return key.toString();
    };
}

注意:如果配置了KeyGenerator ,在进行缓存的时候如果不指定key的话,最后会把生成的key缓存起来,如果同时配置了KeyGenerator 和key则优先使用key。

  1. 在controller或者service的类上面添加@CacheConfig,注解里面的参数详情见下表:

    参数名参数值作用
    cacheNames可以随意填写,一般是一个模块或者一个很重要的功能名称无具体作用,只是用来区分缓存,方便管理
    keyGenerator就是自己配置的KeyGenerator的名称全局key都会以他的策略去生成
    cacheManager自己配置的CacheManager用来操作Cache对象的,很多对于缓存的配置也由他去管理
    cacheResolver
  2. 在标有@CacheConfig的类里面编写一个查询单个对象的方法并添加@Cacheable注解

    @Cacheable(key = "#id", unless = "#result == null") 
    @PatchMapping("/course/{id}")
    public Course courseInfo(@PathVariable Integer id) {
        log.info("进来了 .. ");
        return courseService.getCourseInfo(id);
    }
    

    执行完该方法后,执行结果将会被缓存到Redis:

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-sB1N9TrH-1587032886511)(C:\Users\m1523\AppData\Roaming\Typora\typora-user-images\1578670221953.png)]

    @Cacheable注解中参数详情见下表:

    参数名作用
    cacheNames
    key这里的key的优先级是最高的,可以覆盖掉全局配置的key,如果不配置的话使用的就是全局的key
    keyGenerator
    cacheManager
    condition满足什么条件会进行缓存,里面可以写简单的表达式进行逻辑判断
    unless满足什么条件不进行缓存,里面可以写简单的表达式进行逻辑判断
    sync加入缓存的这个操作是否是同步的

Ⅱ、 修改

  1. 编写一个修改的方法,参数传对象,返回值也改成这个对象

    @PutMapping("/course")
    public Course modifyCoruse(@RequestBody Course course) {
        courseService.updateCourse(course);
        return course;
    }
    
  2. 在方法上面添加 @CachePut(key = "#course.id") 注解,这个注解表示将方法的返回值更新到缓存中,注解中的参数和 @Cacheable 中的一样,这里就略过了。

Ⅲ、 删除

  1. 编写删除方法,在方法上添加@CacheEvict 注解

    @CacheEvict(key = "#id")
    @DeleteMapping("/course/{id}")
    public void removeCourse(@PathVariable Integer id) {
        courseService.remove(id);
    }
    

    CacheEvict 的参数信息见下表:

    参数名描述
    allEntries是否删除该命名空间下面的全部缓存,默认是false
    beforeInvocation在执行删除方法前就执行清空缓存操作,默认是false,如果删除方法执行报错该注解则不执行

三、 基于代码的cache的使用

因为我们有配置的CacheManager,所以可以利用RedisCacheManager对象去手动操作cache,首先将CacheManager注入进来:

@Resource 
private CacheManager cacheManager;

@PatchMapping("/course2/{id}")
public Course course2(@PathVariable Integer id) {
    // 获取指定命名空间的cache
    Cache cache = cacheManager.getCache("course");
    // 通过key获取对应的value
    Cache.ValueWrapper wrapper = cache.get(2);
    if (wrapper == null) {
        // 查询数据库
        Course course = courseService.getCourseInfo(id);
        // 加入缓存
        cache.put(course.getId(), course);
        return course;
    } else {
        // 将缓存的结果返回
        // 因为配置了enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        // 所以在进行强转的时候不会报错
        return (Course) wrapper.get(); 
    }
}

如果还看不明白,请去码云拉取源码 https://gitee.com/tianmaoliln/Spring-Boot-Cache.git

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值