为什么要在应用程序中添加缓存在深入探讨如何向应用程序添加缓存之前,首先想到的问题是为什么我们需要在应用程序中使用缓存。
假设有一个包含客户数据的应用程序,用户发出两个请求来获取客户的数据(id=100)。
这就是没有缓存时的情况。
如您所见,对于每个请求,应用程序都会转到数据库获取数据。从数据库获取数据是一项成本高昂的操作,因为它涉及io。
但是,如果中间有一个缓存存储,可以在其中临时存储短时间的数据,则可以将这些往返保存到数据库并在io时间保存。
这就是使用缓存时上述交互的样子。
在spring boot应用程序中实现缓存springboot提供了什么缓存支持?springboot只提供了一个缓存抽象,您可以使用它将缓存透明、轻松地添加到spring应用程序中。
它不提供实际的缓存存储。
但是,它可以与不同类型的缓存提供程序一起工作,如ehcache、hazelcast、redis、caffee等。
springboot的缓存抽象可以添加到方法中(使用注释)
基本上,在执行方法之前,spring框架将检查方法数据是否已经缓存
如果是,则它将从缓存中获取数据。
否则它将执行该方法并缓存数据
它还提供了从缓存中更新或删除数据的抽象。
在我们当前的博客中,我们将了解如何使用caffeine添加缓存,caffeine是一种基于java8的高性能、接近最优的缓存库。
您可以在 application.yaml 文件中指定使用哪个缓存提供程序来设置 spring.cache.type 属性。
但是,如果没有提供属性,spring将根据添加的库自动检测缓存提供程序。
添加生成依赖项现在假设您已经启动并运行了基本的spring boot应用程序,让我们添加缓存依赖项。
打开 build.gradle 文件,并添加以下依赖项以启用spring boot的缓存
compile('org.springframework.boot:spring-boot-starter-cache')
接下来我们将添加对caffeine的依赖
compile group: 'com.github.ben-manes.caffeine', name: 'caffeine', version: '2.8.5'
缓存配置现在我们需要在spring boot应用程序中启用缓存。
为此,我们需要创建一个配置类并提供注释 @enablecaching 。
@configuration@enablecachingpublic class cacheconfig { }
现在这个类是一个空类,但是我们可以向它添加更多配置(如果需要)。
现在我们已经启用了缓存,让我们提供缓存名称和缓存属性的配置,如缓存大小、缓存过期时间等
最简单的方法是在 application.yaml 中添加配置
spring: cache: cache-names: customers, users, roles caffeine: spec: maximumsize=500, expireafteraccess=60s
上述配置执行以下操作
将可用缓存名称限制为客户、用户和角色。将最大缓存大小设置为500。
当缓存中的对象数达到此限制时,将根据缓存逐出策略从缓存中删除对象。将缓存过期时间设置为1分钟。
这意味着项目将在添加到缓存1分钟后从缓存中删除。
还有另一种配置缓存的方法,而不是在 application.yaml 文件中配置缓存。
您可以在缓存配置类中添加并提供一个 cachemanager bean,该bean可以完成与上面在 application.yaml 中的配置完全相同的工作
@beanpublic cachemanager cachemanager() { caffeine<object, object> caffeinecachebuilder = caffeine.newbuilder() .maximumsize(500) .expireafteraccess( 1, timeunit.minutes); caffeinecachemanager cachemanager = new caffeinecachemanager( "customers", "roles", "users"); cachemanager.setcaffeine(caffeinecachebuilder); return cachemanager;}
在我们的代码示例中,我们将使用java配置。
我们可以在java中做更多的事情,比如配置 removallistener ,当一个项从缓存中删除时执行 removallistener ,或者启用缓存统计记录,等等。
缓存方法结果在我们使用的示例spring boot应用程序中,我们已经有了以下api get /api/v1/customer/{id} 来检索客户记录。
我们将向customerservice类的 getcustomerbyd(longcustomerid) 方法添加缓存。
要做到这一点,我们只需要做两件事
1. 将注释 @cacheconfig(cachenames=“customers”) 添加到 customerservice 类
提供此选项将确保 customerservice 的所有可缓存方法都将使用缓存名称“customers”
2. 向方法 optional getcustomerbyid(long customerid) 添加注释 @cacheable
@service@log4j2@cacheconfig(cachenames = "customers")public class customerservice { @autowired private customerrepository customerrepository; @cacheable public optional<customer> getcustomerbyid(long customerid) { log.info("fetching customer by id: {}", customerid); return customerrepository.findbyid(customerid); }}
另外,在方法 getcustomerbyid() 中添加一个 logger 语句,以便我们知道服务方法是否得到执行,或者值是否从缓存返回。
代码如下:log.info("fetching customer by id: {}", customerid);
测试缓存是否正常工作这就是缓存工作所需的全部内容。现在是测试缓存的时候了。
启动您的应用程序,并点击客户获取url
http://localhost:8080/api/v1/customer/
在第一次api调用之后,您将在日志中看到以下行—“ fetching customer by id ”。
但是,如果再次点击api,您将不会在日志中看到任何内容。这意味着该方法没有得到执行,并且从缓存返回客户记录。
现在等待一分钟(因为缓存过期时间设置为1分钟)。
一分钟后再次点击getapi,您将看到下面的语句再次被记录——“通过id获取客户”。
这意味着客户记录在1分钟后从缓存中删除,必须再次从数据库中获取。
为什么缓存有时会很危险缓存更新/失效通常我们缓存 get 调用,以提高性能。
但我们需要非常小心的是缓存对象的更新/删除。
@cacheput@cacheexecute
如果未将 @cacheput/@cacheexecute 放入更新/删除方法中,get调用中缓存返回的对象将与数据库中存储的对象不同。考虑下面的示例场景。
如您所见,第二个请求已将人名更新为“ john smith ”。但由于它没有更新缓存,因此从此处开始的所有请求都将从缓存中获取过时的个人记录(“ john doe ”),直到该项在缓存中被删除/更新。
缓存复制大多数现代web应用程序通常有多个应用程序节点,并且在大多数情况下都有一个负载平衡器,可以将用户请求重定向到一个可用的应用程序节点。
这种类型的部署为应用程序提供了可伸缩性,任何用户请求都可以由任何一个可用的应用程序节点提供服务。
在这些分布式环境(具有多个应用服务器节点)中,缓存可以通过两种方式实现
应用服务器中的嵌入式缓存(正如我们现在看到的)
远程缓存服务器
嵌入式缓存嵌入式缓存驻留在应用程序服务器中,它随应用程序服务器启动/停止。由于每台服务器都有自己的缓存副本,因此对其缓存的任何更改/更新都不会自动反映在其他应用程序服务器的缓存中。
考虑具有嵌入式缓存的多节点应用服务器的下面场景,其中用户可以根据应用服务器为其请求服务而得到不同的结果。
正如您在上面的示例中所看到的,更新请求更新了 application node2 的数据库和嵌入式缓存。
但是, application node1 的嵌入式缓存未更新,并且包含过时数据。因此, application node1 的任何请求都将继续服务于旧数据。
要解决这个问题,您需要实现 cache replication —其中任何一个缓存中的任何更新都会自动复制到其他缓存(下图中显示为蓝色虚线)
远程缓存服务器解决上述问题的另一种方法是使用远程缓存服务器(如下所示)。
然而,这种方法的最大缺点是增加了响应时间——这是由于从远程缓存服务器获取数据时的网络延迟(与内存缓存相比)
缓存自定义到目前为止,我们看到的缓存示例是向应用程序添加基本缓存所需的唯一代码。
然而,现实世界的场景可能不是那么简单,可能需要进行一些定制。在本节中,我们将看到几个这样的例子
缓存密钥我们知道缓存是密钥、值对的存储。
示例1:默认缓存键–具有单参数的方法
最简单的缓存键是当方法只有一个参数,并且该参数成为缓存键时。在下面的示例中, long customerid 是缓存键
示例2:默认缓存键–具有多个参数的方法
在下面的示例中,缓存键是所有三个参数的simplekey– countryid 、 regionid 、 personid 。
示例3:自定义缓存密钥
在下面的示例中,我们将此人的 emailaddress 指定为缓存的密钥
示例4:使用 keygenerator 的自定义缓存密钥
让我们看看下面的示例–如果要缓存当前登录用户的所有角色,该怎么办。
该方法中没有提供任何参数,该方法在内部获取当前登录用户并返回其角色。
为了实现这个需求,我们需要创建一个如下所示的自定义密钥生成器
然后我们可以在我们的方法中使用这个键生成器,如下所示。
条件缓存在某些用例中,我们只希望在满足某些条件的情况下缓存结果
示例1(支持 java.util.optional –仅当存在时才缓存)
仅当结果中存在 person 对象时,才缓存 person 对象。
@cacheable( value = "persons", unless = "#result?.id")public optional<person> getperson(long personid)
示例2(如果需要,by-pass缓存)
@cacheable(value = "persons", condition="#fetchfromcache")public optional<person> getperson(long personid, boolean fetchfromcache)
仅当方法参数“ fetchfromcache ”为true时,才从缓存中获取人员。通过这种方式,方法的调用方有时可以决定绕过缓存并直接从数据库获取值。
示例3(基于对象属性的条件计算)
仅当价格低于500且产品有库存时,才缓存产品。
@cacheable( value="products", condition="#product.price<500", unless="#result.outofstock")public product findproduct(product product)
@cacheput我们已经看到 @cacheable 用于将项目放入缓存。
但是,如果该对象被更新,并且我们想要更新缓存,该怎么办?
我们已经在前面的一节中看到,不更新缓存post任何更新操作都可能导致从缓存返回错误的结果。
@cacheput(key = "#person.id")public person update(person person)
但是如果 @cacheable 和 @cacheput 都将一个项目放入缓存,它们之间有什么区别?
主要区别在于实际的方法执行
@cacheable@cacheput
缓存失效缓存失效与将对象放入缓存一样重要。
当我们想要从缓存中删除一个或多个对象时,有很多场景。让我们看一些例子。
例1
假设我们有一个用于批量导入个人记录的api。
我们希望在调用此方法之前,应该清除整个 person 缓存(因为大多数 person 记录可能会在导入时更新,而缓存可能会过时)。我们可以这样做如下
@cacheevict( value = "persons", allentries = true, beforeinvocation = true)public void importpersons()
例2
我们有一个delete person api,我们希望它在删除时也能从缓存中删除 person 记录。
@cacheevict( value = "persons", key = "#person.emailaddress")public void deleteperson(person person)
默认情况下 @cacheevict 在方法调用后运行。
以上就是springboot怎么使用caffeine实现缓存的详细内容。