Spring Cache-缓存注解(二)

导读

Spring-Cache手札

Spring Cache抽象-缓存注解

实战-Redis-20Spring缓存机制整合Redis

关于Spring Cache以及注解,之前总结了几篇。现在我们再来细化下


概述

注解描述
@Cacheable表明在进入方法之前, Spring 会先去缓存服务器中查找对应 key 的缓存值,如果找到缓存值,那么 Spring 将不会再调用方法,而是将缓存值读出,返回给调用者;如果没有找到缓存值,那么 Spring 就会执行你的方法,将最后的结果通过 key 保存到缓存服务器中
@CachePutSpring 会将该方法返回的值缓存到缓存服务器中,这里简要注意的是, Spring 不会事先去缓存服务器中查找,而是直接执行方法,然后缓存。换句话说,该方法始终会被 Spring 所调用
@CacheEvict移除缓存对应的 key 的值
@Caching这是一个分组注解,它能够同时应用于其他缓存的注解
  • 注解@Cacheable 和@CachePut 都可以保存缓存键值对,只是它们的方式略有不同, 请注意二者的区别,它们只能运用于有返回值的方法中
  • 而删除缓存 key 的@CacheEvict 则可以用在 void 的方法上,因为它并不需要去保存任何值 。
  • 上述注解都能标注到类或者方法之上,如果放到类上,则对所有的方法都有效,如果放到方法上,则只是对方法有效。
  • 在大部分情况下,会放置到方法上。 @Cacheable 和 @CachePut 可以配置的属性接近。
  • 一般而言,对于查询,我们会考虑使用@Cacheable
  • 对于插入和修改,考虑使用@CachePut
  • 对于删除操作,我们会考虑使用@CacheEvict。

注解@Cacheable 和@CachePut

因为@Cacheable 和@CachePut 两个注解的配置项 比较接近,所以这里就将这两个注解一并来看

属性配置类型说明
valueString[]使用缓存的名称
conditionStringSpring 表达式,如果表达式返回值为 false,则不会将缓存应用到方法上, true 则会
keyStringSpring 表达式,可以通过它来计算对应缓存的 key
unlessStringSpring 表达式,如果表达式返回值为 true,则不会将方法的结果放到缓存上

value 和 key 这两个属性使用得最多,所以先来讨论这两个属性。

value 是一个数组,可以引用多个缓存管理器.

案例----->https://blog.csdn.net/yangshangwei/article/details/82961772#Service_664

在这里插入图片描述

在这里插入图片描述

如上代码所示定义redisCacheManager后就可以引用它了,而对于 key 则是缓存中的键,它支持 Spring 表达式,通过 Spring 表达式就可以自定义缓存的 key。


表达式值的引用

Spring 表达式和缓存注解之间的约定,通过这些约定去引用方法的参数和返回值的内容,使得其注入 key 所定义的 Spring 表达式的结果中。

表达式描述备注
#root.args定义传递给缓存方法的参数不常用,暂不讨论
#root.caches该方法执行是对应的缓存名称,它是一个数组不常用,暂不讨论
#root.target执行缓存的目标对象不常用,暂不讨论
#root.targetClass目标对象的类,它是#root.target.class 的缩写不常用,暂不讨论
#root.method缓存方法不常用,暂不讨论
#root.methodName缓存方法的名称,它是#root.method.name 的缩写不常用,暂不讨论
#result方法返回结果值,还可以使用 Spring 表达式进一步读取其属性请注意该表达式不能用于注解@Cacheable,因为该注解的方法可能不会被执行,这样返回值就无从谈起了
#Argument任意方法的参数,可以通过方法本身的名称或者下标去定义比如 getRole(Long id)方法,想读取 id 这个参数,可以写为#id,或者#a0、#p0,建议写为#id,可读性高

这样就方便使用对应的参数或者返回值作为缓存的 key 了。


注解@CacheEvict

注解@CacheEvict 主要是为了移除缓存对应的键值对,主要对于那些删除的操作,先来了解它存在哪些属性。

属性配置类型说明
valueString[]要使用缓存的名称
keyStringSpring 表达式,可以通过它来计算对应缓存的 key
conditionStringSpring 表达式,如果表达式返回值为 false,则不会将缓存应用到方法上, true 则会
allEntriesboolean如果为 true,则删除特定缓存所有键值对,默认值为 false,请注意它将消除所有缓存服务器的缓存,这个属性慎用
beforelnvocationboolean指定在方法前后移除缓存,如果指定为 true,则在方法前删除缓存:如果为 false,则在方法调用后删除级存,默认值为 false
  • value 和 key 与之前的@Cacheable 和@CachePut 是一致的。
  • 属性 allEntries 要求删除缓存服务器中所有的缓存,这个时候指定的 key 将不会生效,所以这个属性要慎用
  • beforeInvocation 属性指定缓存在方法前或者方法后移除。

案例

Spring缓存机制整合Redis

package com.artisan.ssm_redis.service.impl;

import java.util.List;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cache.annotation.CacheEvict;
import org.springframework.cache.annotation.CachePut;
import org.springframework.cache.annotation.Cacheable;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Isolation;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;

import com.artisan.ssm_redis.dao.RoleDao;
import com.artisan.ssm_redis.domain.Role;
import com.artisan.ssm_redis.service.RoleService;


@Service
public class RoleServiceImpl implements RoleService {

	// 自动注入
	@Autowired
	private RoleDao roleDao;

	/**
	 * 使用@Cacheable定义缓存策略 当缓存中有值,则返回缓存数据,否则访问方法得到数据 通过value引用缓存管理器,通过key定义键
	 * 
	 * @param id
	 *            角色编号
	 * @return 角色
	 */
	@Override
	@Transactional(isolation = Isolation.READ_COMMITTED, propagation = Propagation.REQUIRED)
	@Cacheable(value = "redisCacheManager", key = "'redis_role_'+#id")
	public Role getRole(Long id) {
		return roleDao.getRole(id);
	}

	/**
	 * 使用@CachePut则表示无论如何都会执行方法,最后将方法的返回值再保存到缓存中
	 * 使用在插入数据的地方,则表示保存到数据库后,会同期插入到Redis缓存中
	 * 
	 * @param role
	 *            角色对象
	 * @return 角色对象(会回填主键)
	 */
	@Override
	@Transactional(isolation = Isolation.READ_COMMITTED, propagation = Propagation.REQUIRED)
	@CachePut(value = "redisCacheManager", key = "'redis_role_'+#result.id")
	public Role insertRole(Role role) {
		roleDao.insertRole(role);
		return role;
	}

	/**
	 * 使用@CachePut,表示更新数据库数据的同时,也会同步更新缓存
	 * 
	 * @param role
	 *            角色对象
	 * @return 影响条数
	 */
	@Override
	@Transactional(isolation = Isolation.READ_COMMITTED, propagation = Propagation.REQUIRED)
	@CachePut(value = "redisCacheManager", key = "'redis_role_'+#role.id")
	public int updateRole(Role role) {
		return roleDao.updateRole(role);
	}

	/**
	 * 使用@CacheEvict删除缓存对应的key
	 * 
	 * @param id
	 *            角色编号
	 * @return 返回删除记录数
	 */
	@Override
	@Transactional(isolation = Isolation.READ_COMMITTED, propagation = Propagation.REQUIRED)
	@CacheEvict(value = "redisCacheManager", key = "'redis_role_'+#id")
	public int deleteRole(Long id) {
		return roleDao.deleteRole(id);
	}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

小小工匠

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

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

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

打赏作者

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

抵扣说明:

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

余额充值