springboot 2.4.0中的缓存注解

上一篇笔记有提及并介绍了@EnableCaching与@Cacheable注解,这篇再介绍剩下几个重要的缓存注解。

详细内容(如注解实现原理等)在今后学习过程中再添砖加瓦,互相学习。



spring提供的缓存注解

在这里插入图片描述

springboot提供了几个重要的缓存注解,@EnableCaching、@Cacheable、@CachePut、@CacheEvict、@Caching、@CacheConfig。下面做一一介绍并结合自己的代码记录。

背景

自Spring3.1开始,Spring就自带了对缓存的支持。我们可以直接使用Spring缓存技术将某些数据放入本机的缓存中;Spring缓存技术也可以搭配其他缓存中间件(如Redis等)进行使用,将某些数据写入到缓存中间件(缓存中间件可能在其他机器上)中。

所以需要确认Spring版本不低于3.1

我使用的springboot版本为2.4.0,对应spring版本为5.3.2。

@EnableCaching注解

开关性注解,在项目启动类或某个配置类上使用此注解后,则表示允许使用注解的方式进行缓存操作
在这里插入图片描述

@Cacheable注解

可用于类或方法上,在目标方法执行前,会根据key先去缓存中查询看是否有数据,就直接返回缓存中的key对应的value值。不再执行目标方法;则执行目标方法,并将方法的返回值作为value,并以键值对的形式存入缓存,几个重要属性(condition、unless等)可以查看上一篇笔记。
【springboot2.4.0缓存的工作原理与@Cacheable运行流程-笔记】

/**
     * 将方法的运行结果进行缓存,以后再要相同的数据,直接从缓存中获取,不用调用方法。
     *
     * CacheManager管理多个Cache组件的,对缓存的真正CRUD操作在缓存组件中,每一个缓存组件有自己唯一一个名字
     *      @Cacheable 几个属性
     *          cacheNames/value : 指定缓存组件的名字,将方法的返回结果放在哪个缓存中,是数组的方式,可以指定多个缓存。
     *          key : 缓存数据使用的key,可以用它来指定,默认是使用方法参数的值 1  : 方法的返回值
     *              编写SpEl, #id参数id的值 #a0 #p0 #root.args[0]
     *              如果想指定成字符串 --> getEmp[2]
     *          keyGenerrator : key的生成器,也可以自己指定key的生成器的组件id
     *              key / keyGenerrator 二选一使用
     *          cacheManager : 指定缓存管理器, 或者cacheResolver : 指定缓存解析器 二选一使用
     *          condition : 指定符合条件的情况下才缓存
     *              condition = "#id > 0" 如果传入参数 > 0 ,则缓存
     *          unless : 否定缓存,当unless指定的条件为true,方法的返回值就不会被缓存;可以获取到结果进行判断
     *              unless = "#result == null" 如果查询结果为空,则不缓存
     *          sync : 是否使用异步模式
     *
     * 原理 :
     *  1.自动配置类 : CacheAutoConfiguration
     *  2.缓存配置类
     *  org.springframework.boot.autoconfigure.cache.GenericCacheConfiguration
     *  org.springframework.boot.autoconfigure.cache.JCacheCacheConfiguration
     *  org.springframework.boot.autoconfigure.cache.EhCacheCacheConfiguration
     *  org.springframework.boot.autoconfigure.cache.HazelcastCacheConfiguration
     *  org.springframework.boot.autoconfigure.cache.InfinispanCacheConfiguration
     *  org.springframework.boot.autoconfigure.cache.CouchbaseCacheConfiguration
     *  org.springframework.boot.autoconfigure.cache.RedisCacheConfiguration
     *  org.springframework.boot.autoconfigure.cache.CaffeineCacheConfiguration
     *  org.springframework.boot.autoconfigure.cache.SimpleCacheConfiguration
     *  org.springframework.boot.autoconfigure.cache.NoOpCacheConfiguration
     *  3.哪个配置类默认生效 SimpleCacheConfiguration
     *  4.给容器中注册了一个CacheManager : ConcurrentMapCacheManager
     *  5.getCache 双重判断 可以获取和创建ConcurrentMapCache类型的缓存组件;它的作用是将数据保存在ConcurrentMap中。
     *
     *  运行流程 :
     *      @Cacheable
     *      1.方法运行之前,先去查询Cache(缓存组件),按照cacheNames指定的名字获取。 --> 先来到getCache获取缓存 传入参数 name = "emp"
     *          (CacheManager先获取相应的缓存)
     *      第一次拿到的缓存为Cache : null    name : emp
     *        没有的话会自动调用createConcurrentMapCache创建一个缓存。
     *
     *      2.去Cache中查找缓存的内容,使用一个key,默认就是方法的参数
     *      key是按照某种策略生成的 默认是调用context.generateKey 中的ConcurrentMap的实现SimpleKeyGenerator生成key
     *      SimpleKeyGenerator生成key的默认策略
     *          如果没有参数,key = new SimpleKey();
     *          如果有一个参数,key = 参数的值
     *          如果有多个参数,key = new SimpleKey(params);
     *
     *      3.没有查到缓存就调用目标方法
     *
     *      4.将目标方法返回的结果,放进缓存中
     *      ConcurrentMapCache中的put方法 通过store(ConcurrentMap).put(key,value)
     *
     *  @Cacheable 标注的方法执行之前先来检查缓存中有没有这个数据,默认按照参数的值作为key去查询缓存,
     *  如果没有就运行方法,并将结果存入缓存,以后再来调用就可以直接使用缓存中的数据。
     *
     *  核心 :
     *      1.使用CacheManager【ConcurrentMapCacheManager】按照名字得到Cache【ConcurrentMapCache】组件
     *      2.key是使用KeyGenerator生成的,默认是SimpleKeyGenerator
     *
     *
     * @param id
     * @return
     */
    @Cacheable(cacheNames = "emp")
    @Override
    public Employee getEmp(Integer id) {
        System.out.println("查询" + id + "号员工");
        Employee emp = employeeMapper.getEmpById(id);
        return emp;
    }

@CachePut注解

可用于类或方法上,在执行完目标方法后,并将方法的返回值作为value,并以键值对的形式存入缓存中。
简单来说,就是
说法① 既调用方法,又更新缓存数据
说法② 修改了数据库某个数据,同时更新缓存

对比 @Cacheable@CachePut 的运行时机 :
@Cacheable 是在调用方法之前,查询缓存,如果有则直接返回缓存的数据,如果没有则运行方法之后,将返回结果缓存。
@CachePut 是直接先调用方法,调用完方法再把方法的结果放入缓存中。

几个注意点:

  • @Cacheable 的key是不能用 #result 因为它在查询缓存之后。
  • 保存的时候要注意指定一下key,在@Cacheable如果默认使用传参id作为key,那么在@CacheOut中可以通过 #result.id 来设置key。
	/**
	*  测试步骤:
	*      1.查询1号员工,查到的结果会放在缓存中。
	*          key = 1,value = 之前的数据
	*      2.以后查询还是之前的结果
	*      3.更新1号员工【id=1, lastName=xiaopangzi, email=null, gender=1, dId=1】
	*          将方法的返回值也放进缓存了
	*          key = 传入的Employee对象 value = 返回的Employee对象
	*      4.查询1号员工
	*          应该是更新后的员工
	*                  key = "#employee.id" --> 使用传入的参数的员工id
	*                  key = "#result.id" --> 使用返回值的id
	*                  @Cacheable 的key是不能用#result 因为它在查询缓存之后
	*              为什么是没更新前的? 【1号员工没有在缓存中更新,key不一样】
	*/
	@CachePut(value = "emp", key = "#result.id")
    public Employee updateEmp(Employee employee) {
        System.out.println("updateEmp:" + employee);
        employeeMapper.updateEmp(employee);
        return employee;
    }

@CacheEvict注解

可用于类或方法上;在执行完目标方法后,清除缓存中对应key的数据(如果缓存中有对应key的数据缓存的话)。

几个属性设置 :

  • key 来指定要清除的数据。
  • allEntries = true 将缓存的所有数据都删除。
  • beforeInvocation = false 默认为false 是否在方法之前执行 默认是在方法执行之后执行的,如果方法在执行时出现异常,缓存就不会被清除。beforeInvocation = true时,代表不管方法执行成功或失败,都会清除缓存。
	/**
	     * @CacheEvict : 缓存清除
	     * key 指定要清除的数据
	     * allEntries = true 将缓存的所有数据都删除
	     * beforeInvocation = false 默认为false 是否在方法之前执行 默认是在方法执行之后执行的
	     *  如果方法在执行时出现异常,缓存就不会被清除
	     *
	     *  beforeInvocation = true时,代表不管方法执行成功或失败,都会清除缓存
	     */
	@CacheEvict(value = "emp", key = "#id")
    public void deleteEmp(Integer id) {
        System.out.println("删除" + id + "号员工");
        employeeMapper.deleteEmp(id);
    }

@Caching注解

此注解即可作为@Cacheable、@CacheEvict、@CachePut三种注解中的的任何一种或几种来使用。一般用于定义复杂的缓存规则。

	@Caching(
            cacheable = {
                    @Cacheable(value = "emp", key = "#lastName")
            },
            put = {
                    @CachePut(value = "emp", key = "#result.id"),
                    @CachePut(value = "emp", key = "#result.email")
            }
    )
    public Employee getEmpByLastName(String lastName) {
        Employee employee = employeeMapper.getEmpByLastName(lastName);
        return employee;
    }

@CacheConfig注解

@Cacheable、@CacheEvict、@CachePut这三个注解的cacheNames属性是必填项(或value属性是必填项,因为value属性是cacheNames的别名属性);如果上述三种注解都用的是同一个cacheNames的话,那么在每此都写cacheNames的话,就会显得麻烦。如将@CacheConfig注解就是来配置一些公共属性(如:cacheNames、keyGenerator等)的值的。

简单来说,就是抽取缓存的公共配置

加在类头。

	@CacheConfig(cacheNames = "emp")
	@Service
	@Transactional
	public class EmployeeServiceImpl implements EmployeeService {
		...
	}

结语

如有不正之处,欢迎指正,互相学习,多加借鉴。

  • ٩꒰▽ ꒱۶⁼³₌₃ 学习去咯
  • …φ(๑˃∀˂๑)♪ 学习是我的全部
  • (っ•̀ω•́)っ✎⁾⁾ 我爱学习
  • :ஐ٩(๑´ᵕ`)۶ஐ: 学习使我进步
  • *✧⁺˚⁺ପ(๑・ω・)੭ु⁾⁾ 好好学习天天向上
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

舍其小伙伴

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值