- 作者简介
- 内容提要
- 关于本书
- 路线图
- 代码规范与下载
- 作者在线
- 封面插图简介
- 前言
- 译者序
- 致谢
- 第1部分 Spring 的核心
- 第1章 Spring 之旅
- 第2章 装配 Bean
- 第3章 高级装配
- 第4章 面向切面的 Spring
- 第2部分 Web 中的 Spring
- 第5章 构建 Spring Web 应用程序
- 第6章 渲染 Web 视图
- 第7章 Spring MVC 的高级技术
- 第8章 使用 Spring Web Flow
- 第9章 保护 Web 应用
- 第3部分 后端中的 Spring
- 第10章 通过 Spring 和 JDBC 征服数据库
- 第11章 使用对象-关系映射持久化数据
- 第12章 使用 NoSQL 数据库
- 第13章 缓存数据
- 第14章 保护方法应用
- 第4部分 Spring 集成
- 第15章 使用远程服务
- 第16章 使用 Spring MVC 创建 REST API
- 第17章 Spring消息
- 第18章 使用 WebSocket 和 STOMP 实现消息功能
- 第19章 使用 Spring 发送 Email
- 第20章 使用 JMX 管理 Spring Bean
- 第21章 借助 Spring Boot 简化 Spring 开发
13.2.1 填充缓存
我们可以看到,@Cacheable和@CachePut注解都可以填充缓存,但是它们的工作方式略有差异。
@Cacheable首先在缓存中查找条目,如果找到了匹配的条目,那么就不会对方法进行调用了。如果没有找到匹配的条目,方法会被调用并且返回值要放到缓存之中。而@CachePut并不会在缓存中检查匹配的值,目标方法总是会被调用,并将返回值添加到缓存之中。
@Cacheable和@CachePut有一些属性是共有的,参见表13.2。
表13.2 @Cacheable和@CachePut有一些共有的属性
属 性 | 类 型 | 描 述 |
value | String[] | 要使用的缓存名称 |
condition | String | SpEL表达式,如果得到的值是false的话,不会将缓存应用到方法调用上 |
key | String | SpEL表达式,用来计算自定义的缓存key |
unless | String | SpEL表达式,如果得到的值是true的话,返回值不会放到缓存之中 |
在最简单的情况下,在@Cacheable和@CachePut的这些属性中,只需使用value属性指定一个或多个缓存即可。例如,考虑SpittleRepository的findOne()方法。在初始保存之后,Spittle就不会再发生变化了。如果有的Spittle比较热门并且会被频繁请求,反复地在数据库中进行获取是对时间和资源的浪费。通过在findOne()方法上添加@Cacheable注解,如下面的程序清单所示,能够确保将Spittle保存在缓存中,从而避免对数据库的不必要访问。
程序清单13.6 通过使用@Cacheable,在缓存中存储和获取值
当findOne()被调用时,缓存切面会拦截调用并在缓存中查找之前以名spittleCache存储的返回值。缓存的key是传递到findOne()方法中的id参数。如果按照这个key能够找到值的话,就会返回找到的值,方法不会再被调用。如果没有找到值的话,那么就会调用这个方法,并将返回值放到缓存之中,为下一次调用findOne()方法做好准备。
在程序清单13.6中,@Cacheable注解被放到了JdbcSpittleRepository的findOne()方法实现上。这样能够起作用,但是缓存的作用只限于JdbcSpittleRepository这个实现类中,SpittleRepository的其他实现并没有缓存功能,除非也为其添加上@Cacheable注解。因此,可以考虑将注解添加到SpittleRepository的方法声明上,而不是放在实现类中:
当为接口方法添加注解后,@Cacheable注解会被SpittleRepository的所有实现继承,这些实现类都会应用相同的缓存规则。
将值放到缓存之中
@Cacheable会条件性地触发对方法的调用,这取决于缓存中是不是已经有了所需要的值,对于所注解的方法,@CachePut采用了一种更为直接的流程。带有@CachePut注解的方法始终都会被调用,而且它的返回值也会放到缓存中。这提供一种很便利的机制,能够让我们在请求之前预先加载缓存。
例如,当一个全新的Spittle通过SpittleRepository的save()方法保存之后,很可能马上就会请求这条记录。所以,当save()方法调用后,立即将Spittle塞到缓存之中是很有意义的,这样当其他人通过findOne()对其进行查找时,它就已经准备就绪了。为了实现这一点,可以在save()方法上添加@CachePut注解,如下所示:
当save()方法被调用时,它首先会做所有必要的事情来保存Spittle,然后返回的Spittle会被放到spittleCache缓存中。
在这里只有一个问题:缓存的key。如前文所述,默认的缓存key要基于方法的参数来确定。因为save()方法的唯一参数就是Spittle,所以它会用作缓存的key。将Spittle放在缓存中,而它的缓存key恰好是同一个Spittle,这是不是有一点诡异呢?
显然,在这个场景中,默认的缓存key并不是我们想要的。我们需要的缓存key是新保存Spittle的ID,而不是Spittle本身。所以,在这里需要指定一个key而不是使用默认的key。让我们看一下怎样自定义缓存key。
自定义缓存key
@Cacheable和@CachePut都有一个名为key属性,这个属性能够替换默认的key,它是通过一个SpEL表达式计算得到的。任意的SpEL表达式都是可行的,但是更常见的场景是所定义的表达式与存储在缓存中的值有关,据此计算得到key。
具体到我们这个场景,我们需要将key设置为所保存Spittle的ID。以参数形式传递给save()的Spittle还没有保存,因此并没有ID。我们只能通过save()返回的Spittle得到id属性。
幸好,在为缓存编写SpEL表达式的时候,Spring暴露了一些很有用的元数据。表13.3列出了SpEL中可用的缓存元数据。
表13.3 Spring提供了多个用来定义缓存规则的SpEL扩展
表 达 式 | 描 述 |
#root.args | 传递给缓存方法的参数,形式为数组 |
#root.caches | 该方法执行时所对应的缓存,形式为数组 |
#root.target | 目标对象 |
#root.targetClass | 目标对象的类,是#root.target.class的简写形式 |
#root.method | 缓存方法 |
#root.methodName | 缓存方法的名字,是#root.method.name的简写形式 |
#result | 方法调用的返回值(不能用在@Cacheable注解上) |
#Argument | 任意的方法参数名(如#argName)或参数索引(如#a0或#p0) |
对于save()方法来说,我们需要的键是所返回Spittle对象的id属性。表达式#result能够得到返回的Spittle。借助这个对象,我们可以通过将key属性设置为#result.id来引用id属性:
按照这种方式配置@CachePut,缓存不会去干涉save()方法的执行,但是返回的Spittle将会保存在缓存中,并且缓存的key与Spittle的id属性相同。
条件化缓存
通过为方法添加Spring的缓存注解,Spring就会围绕着这个方法创建一个缓存切面。但是,在有些场景下我们可能希望将缓存功能关闭。
@Cacheable和@CachePut提供了两个属性用以实现条件化缓存:unless和condition,这两个属性都接受一个SpEL表达式。如果unless属性的SpEL表达式计算结果为true,那么缓存方法返回的数据就不会放到缓存中。与之类似,如果condition属性的SpEL表达式计算结果为false,那么对于这个方法缓存就会被禁用掉。
表面上来看,unless和condition属性做的是相同的事情。但是,这里有一点细微的差别。unless属性只能阻止将对象放进缓存,但是在这个方法调用的时候,依然会去缓存中进行查找,如果找到了匹配的值,就会返回找到的值。与之不同,如果condition的表达式计算结果为false,那么在这个方法调用的过程中,缓存是被禁用的。就是说,不会去缓存进行查找,同时返回值也不会放进缓存中。
作为样例(尽管有些牵强),假设对于message属性包含“NoCache”的Spittle对象,我们不想对其进行缓存。为了阻止这样的Spittle对象被缓存起来,可以这样设置unless属性:
为unless设置的SpEL表达式会检查返回的Spittle对象(在表达式中通过#result来识别)的message属性。如果它包含“NoCache”文本内容,那么这个表达式的计算值为true,这个Spittle对象不会放进缓存中。否则的话,表达式的计算结果为false,无法满足unless的条件,这个Spittle对象会被缓存。
属性unless能够阻止将值写入到缓存中,但是有时候我们希望将缓存全部禁用。也就是说,在一定的条件下,我们既不希望将值添加到缓存中,也不希望从缓存中获取数据。
例如,对于ID值小于10的Spittle对象,我们不希望对其使用缓存。在这种场景下,这些Spittle是用来进行调试的测试条目,对其进行缓存并没有实际的价值。为了要对ID小于10的Spittle关闭缓存,可以在@Cacheable上使用condition属性,如下所示:
如果findOne()调用时,参数值小于10,那么将不会在缓存中进行查找,返回的Spittle也不会放进缓存中,就像这个方法没有添加@Cacheable注解一样。
如样例所示,unless属性的表达式能够通过#result引用返回值。这是很有用的,这么做之所以可行是因为unless属性只有在缓存方法有返回值时才开始发挥作用。而condition肩负着在方法上禁用缓存的任务,因此它不能等到方法返回时再确定是否该关闭缓存。这意味着它的表达式必须要在进入方法时进行计算,所以我们不能通过#result引用返回值。
我们现在已经在缓存中添加了内容,但是这些内容能被移除掉吗?接下来看一下如何借助@CacheEvict将缓存数据移除掉。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论