Spring基于注解的缓存声明深入探究
作者:凌波漫步& 发布时间:2023-01-20 13:26:06
一、概述
从3.1版本起,Spring框架就已经支持将缓存添加到现有的Spring应用中,和事务支持一样,缓存抽象允许在对代码影响最小的情况下一致性地使用各种缓存解决方案。
从Spring 4.1版本起,有了JSR-107
注解和更多定制化的选项支持后,缓存抽象有了重大的改进。
二、声明式基于注解的缓存
对于缓存声明,该抽象提供了一套Java注解:
@Cacheable
:触发缓存构建。@CacheEvict
:触发缓存销毁。@CachePut
:更新缓存。@Caching
:重组应用到方法上的多个缓存操作。@CacheConfig
:类级别共享缓存相关的通用设置。
1、@Cacheable注解
正如其名,@Cacheable
注解用来区分方法执行结果是否应该被缓存,如果后续该方法再次被调用,方法的执行结果直接从缓存中获取,而不会调用实际的方法逻辑。示例如下:
@Cacheable("books")
public Book findBook(ISBN isbn) {...}
当然我们也可以指定多个缓存名称,如果至少一个缓存被命中,那么关联的缓存结果就会返回,示例如下:
@Cacheable({"books", "isbns"})
public Book findBook(ISBN isbn) {...}
(1) 默认缓存key的生成
因为缓存都是key-value存储,每次缓存方法的调用都会被转义为缓存key的访问。Spring缓存抽象对于key的生成会采用KeyGenerator
来生成,算法如下:
如果没有方法参数,返回
SimpleKey.EMPTY
。如果该方法只有一个参数,返回参数实例。
如果方法不止一个参数,返回包含所有参数的
SimpleKey
实例。
这种key的生成策略适用于大部分场景,只要方法参数合理实现了hashCode()
和equals()
方法。
SimpleKeyGenerator
源码如下:
public class SimpleKeyGenerator implements KeyGenerator {
@Override
public Object generate(Object target, Method method, Object... params) {
return generateKey(params);
}
/**
* Generate a key based on the specified parameters.
*/
public static Object generateKey(Object... params) {
if (params.length == 0) {
return SimpleKey.EMPTY;
}
if (params.length == 1) {
Object param = params[0];
if (param != null && !param.getClass().isArray()) {
return param;
}
}
return new SimpleKey(params);
}
}
备注:如果要实现自定义key生成策略,需要实现org.springframework.cache.interceptor.KeyGenerator
接口。
(2) 声明式自定义key生成
目标方法可能会有多个参数,有些参数可能只应用于方法逻辑,而不适合用作key的生成,例如:
@Cacheable("books")
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed)
对于这种情况,@Cacheable
注解有一个key
属性,通过该属性可以自定义Key生成。我们也可以使用SPEL(Spring表达式语言)去指定参数或者参数的嵌套属性,示例如下:
@Cacheable(cacheNames="books", key="#isbn")
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed)
@Cacheable(cacheNames="books", key="#isbn.rawNumber")
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed)
@Cacheable(cacheNames="books", key="T(someType).hash(#isbn)")
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed)
当然,我们也可以通过@Cacheable
注解指定自定义的KeyGenerator
实例,示例如下:
@Cacheable(cacheNames="books", keyGenerator="myKeyGenerator")
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed)
备注:key
和keyGenerator
参数是互斥的,如果两者都指定会触发异常。
(3) 默认缓存解析
Spring缓存抽象通过CacheResolver
去解析操作级别的缓存,而CacheResolver
会用CacheManager
去获取缓存,接口定义如下:
@FunctionalInterface
public interface CacheResolver {
/**
* Return the cache(s) to use for the specified invocation.
* @param context the context of the particular invocation
* @return the cache(s) to use (never {@code null})
* @throws IllegalStateException if cache resolution failed
*/
Collection<? extends Cache> resolveCaches(CacheOperationInvocationContext<?> context);
}
public interface CacheManager {
/**
* Get the cache associated with the given name.
* <p>Note that the cache may be lazily created at runtime if the
* native provider supports it.
* @param name the cache identifier (must not be {@code null})
* @return the associated cache, or {@code null} if such a cache
* does not exist or could be not created
*/
@Nullable
Cache getCache(String name);
/**
* Get a collection of the cache names known by this manager.
* @return the names of all caches known by the cache manager
*/
Collection<String> getCacheNames();
}
(4) 自定义缓存解析
默认缓存解析对于单CacheManager
应用适应很好,对于有多个缓存管理器的应用,我们可以对每个操作设置缓存管理器,如下:
@Cacheable(cacheNames="books", cacheManager="anotherCacheManager")
public Book findBook(ISBN isbn) {...}
(5) 条件式缓存
有时方法缓存结果可能要取决于指定的参数,缓存注解通过支持SPEL
的condition
属性实现该功能,示例如下:
@Cacheable(cacheNames="book", condition="#name.length() < 32")
public Book findBook(String name)
备注:只有当参数name的长度小于32时,方法结果才会被缓存。
除了condition
属性,unless
属性可以用来决定方法返回值不缓存。与condition
不同,unless
表达式在方法被调用后才会执行,示例如下:
@Cacheable(cacheNames="book", condition="#name.length() < 32", unless="#result.hardback")
public Book findBook(String name)
备注:如果Book对象的hardback
属性为true则不缓存,为false才缓存。
当然,缓存抽象同时也支持java.util.Optional
,只有当Optional
中的值存在时,方法返回值才会被缓存。#result
代表方法的执行结果,上面的我们可以改写:
@Cacheable(cacheNames="book", condition="#name.length() < 32", unless="#result?.hardback")
public Optional<Book> findBook(String name)
备注;#result
引用的始终是Book对象,而不是Optional对象,因为返回值可能为空,所以我们应该使用安全导航操作符 => ?.
关于其它可以用的缓存SpEL表达式上下文,可以参考:Available Caching SpEL Evaluation Context。
2、@CachePut注解
这个注解主要用于更新缓存,也就说带有该注解的方法总是会执行,并且方法的返回值会刷新缓存。该注解和@Cacheable
的参数相同,示例如下:
@CachePut(cacheNames="book", key="#isbn")
public Book updateBook(ISBN isbn, BookDescriptor descriptor)
备注:@CachePut
和@Cacheable
的主要区别在于后者会通过缓存跳过方法的执行,而前者为了更新缓存会迫使方法执行。
3、@CacheEvict注解
这个注解主要用来清除缓存,与@Cacheable
注解相反,方法的执行会触发从缓存中删除数据。@CacheEvit
注解要求指定一个或多个缓存名。除此之外,该注解还有一个额外的属性allEntries
,指定该属性值为true后会清除某个缓存名下的所有缓存key。示例如下:
@CacheEvict(cacheNames="books", allEntries=true)
public void loadBooks(InputStream batch)
备注:缓存名为books下的所有缓存key都会被清除。
4、@Caching注解
有些情况下,相同类型多个注解,如@CacheEvict
或者@CachePut
需要被指定。@Caching
注解允许多个嵌套@Cacheable
、@CachePut
、@CacheEvict
注解用在同一个方法上。示例如下:
@Caching(evict = { @CacheEvict("primary"), @CacheEvict(cacheNames="secondary", key="#p0") })
public Book importBooks(String deposit, Date date)
5、@CacheConfig注解
目前我们已经了解到缓存操作提供了很多定制化的选项,然而有些定制化选项如果应用到类中的所有操作可能会有些冗余,示例如下:
@CacheConfig("books")
public class BookRepositoryImpl implements BookRepository {
@Cacheable
public Book findBook(ISBN isbn) {...}
}
@CacheConfig是一个类级别的注解,这个注解可以共享缓存名称,自定义的KeyGenerator
,自定义的CacheManager
和自定义的CacheResolver
。
方法操作级别的自定义选项总是会重写@CacheConfig
中的自定义选项。下面是每个缓存操作自定义选项对应的3个级别,优先级从上至下越来越高。
全局配置的
CacheManager
,KeyGenerator
等。类级别,通过
@CacheConfig
指定。方法操作级别。
三、开启声明式缓存注解
直接在配置类上加上#EnableCaching
即可,如下:
@Configuration
@EnableCaching
public class AppConfig {
}
四、使用自定义注解
Spring缓存抽象允许我们用自定义注解去标识什么方法可以触发缓存构建或者消除。@Cacheable
, @CachePut
, @CacheEvict
and @CacheConfig
这些注解都可以作为元注解,其实即使可以修饰其它注解,示例如下:
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
@Cacheable(cacheNames="books", key="#isbn")
public @interface SlowService {
}
上面我们自定义了SlowService
注解,该注解被@Cacheable
所修饰,现在我们可以用自定义注解代替如下代码:
@Cacheable(cacheNames="books", key="#isbn")
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed)
替代代码如下:
@SlowService
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed)
尽管@SlowService
注解并不是Spring原生注解,但Spring容器会在运行时识别并且知道它是用来干嘛的。
备注:后面我们会利用自定义注解实现自定义过期时间的缓存方案。
来源:https://blog.csdn.net/lingbomanbu_lyl/article/details/126224449


猜你喜欢
- 先上代码,再来分析public class FileDownloadList {/**上下文*/ private Context mCont
- 一、坐标分类地图坐标大致分为几种: 1、GPS、WGS84,也就是原始坐标体系,
- 本文实例讲述了Android实现将应用崩溃信息发送给开发者并重启应用的方法。分享给大家供大家参考,具体如下:在开发过程中,虽然经过测试,但在
- Android中手机震动的设置(Vibrator)的步骤: a、通过系统服务获得手机震动服务,Vibrator vibrator = (Vi
- Dubbo的扩展点加载从JDK标准的SPI(Service Provider Interface)扩展点发现机制加强而来。Dubbo改进了J
- 一、实现了Aware的接口Spring中有很多继承于aware中的接口,这些接口到底是做什么用到的,下面我们就一起来看看吧。Aware 接口
- RandomAccessFileRandomAccessFile是用来访问那些保存数据记录的文件的,你就可以用seek( )方法来访问记录,
- * 其实就是java.lang.reflect.Proxy类动态的根据您指定的所有接口生成一个class byte,该class会继承P
- 最近学习了 C#实现文件上传与下载,现在分享给大家。1、C#文件上传创建MyUpload.htm页面,用于测试<form name=&
- 今天看一个教程,看到一个颜色渐变的ProgressBar,觉得有点意思,所以记录一番。下面这个是效果图颜色渐变的ProgressBar看到效
- 本文实例为大家分享了java自定义异常打印内容的具体代码,供大家参考,具体内容如下背景:在开发中,我们可能会使用到自定义异常,但是,这个自定
- 一、String与Date(java.util.Date)互转 1.1 String -&g
- 我们在使用SpringData JPA框架时,进行条件查询,如果是固定条件的查询,我们可以使用符合框架规则的自定义方法以及@Query注解实
- 目录1.问题引出2.解决办法3.另外一种自定义序列化机制(介绍Externalizable)1.问题引出在某些情况下,我们可能不想对于一个对
- 一、insert1.插入操作public class CRUDTests { @Autowired
- 向量向量是序列容器,表示可以更改大小的数组。就像数组一样,向量对其元素使用连续的存储位置,这意味着也可以使用指向其元素的常规指针上的偏移量来
- 访问修饰符都知道是什么,但是在这之前没有深入的去研究和探索,每天都接触的东西应该清楚才可以。最基础的三个访问修饰符:public 、priv
- Spring是什么?我们通常所说的 Spring 指的是 Spring Framework(Spring 框架),它是⼀个开源框架,有着活跃
- idea乱码修改bin目录下的idea.exe.vmoptions无效今天在学习Activiti工作流的时候,发现创建bpmn文件总是出现中
- 1,MainActivity的xml布局<?xml version="1.0" encoding="ut