接上一节。
@CachePut:既调用方法,又更新缓存。
使用场景:当更改了数据库的某个数据,同时也更新缓存。
运行时机:先调用目标方法,然后将结果放入缓存。
package com.gong.springbootcache.controller; import com.gong.springbootcache.bean.Employee; import com.gong.springbootcache.service.EmployeeService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.cache.annotation.CachePut; import org.springframework.cache.annotation.Cacheable; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PathVariable; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; @Controller public class EmployeeController { @Autowired EmployeeService employeeService; //value:指定缓存的名字,每个缓存组件有一个唯一的名字。缓存组件由CacheManager进行管理。 //key:缓存数据时用到的key,默认使用方法参数的值,1-方法返回值 key = //#id也可这么表示:#root.args[0](第一个参数) //keyGenerator:指定生成缓存的组件id,使用key或keyGenerator其中一个即可 //cacheManager,cacheResolver:指定交由哪个缓存管理器,使用其中一个参数即可 //condition:指定符合条件时才进行缓存 //unless:当unless指定的条件为true,方法的返回值就不会被缓存 //sync:是否使用异步模式 //"#root.methodName+'[+#id+]'" @Cacheable(value = "emp") @ResponseBody @RequestMapping("/emp/{id}") public Employee getEmp(@PathVariable("id") Integer id){ Employee emp = employeeService.getEmp(id); return emp; } @CachePut(value = "emp") @GetMapping("/emp") public Employee updateEmp(Employee employee){ Employee emp = employeeService.updateEmp(employee); return emp; } }
第一次查询:
没有使用缓存。
第二次查询:使用到了缓存,不必再发送sql。
然后进行更新:
先执行了方法,也就是会发送sql:
然后我们再执行一次查询:
此时我们查询出来的是从缓存中获取的,但是,为什么缓存没有进行更新呢?
这是因为getEmp中参数为id,updateEmp参数中为employee。我们知道,如果使用缓存时不指定缓存的key,也就是设置key或者keyGenerator属性,那么,缓存的key就是方法中参数的组合,所以,若果要实现同步的话,我们要指定缓存的key为相同的值。
我们只需要将updateEmp方法中的CachePut注解中加上:key="#employee.id"。
此时再按照之前的步骤再来一次,记得先把2号员工的数据先改回去,在修改后再进行查询:
得到了我们想要的值,同时也没有发送sql。