原文: https://howtodoinjava.com/spring-boot2/spring-boot-cache-example/

在本 Spring Boot 教程中,从 Spring 框架缓存支持中学习轻松管理应用程序缓存。 Spring 在缓存切面具有一些不错的功能, spring cache API 上的抽象非常易于使用。

1. 什么是缓存?

缓存是一种增强系统性能的机制。 它是位于应用程序和持久数据库之间的临时内存。 高速缓存存储器存储最近使用的数据项,以尽可能减少数据库命中的次数。

1.1. 为什么我们需要缓存?

缓存应用程序中经常使用的数据是提高应用程序性能的一种非常流行的技术。 通过缓存,我们将此类经常访问的数据存储在内存中,以避免每次用户请求数据时都访问昂贵的后端。 与从数据库,文件系统或其他服务调用中获取数据相比,从内存中进行数据访问总是更快。

1.2. 应该缓存什么数据?

这主要是关于应该驻留在缓存中并经过缓存生命周期的数据类型的决定。 在不同的情况下以及对我们可以容忍过时的数据的要求切面,它会有所不同。

因此,每个项目的缓存候选者将有所不同,但仍然只是缓存的少数几个示例:

  • 电子商务商店中可用的产品列表
  • 任何不经常更改的主数据
  • 任何经常使用的数据库读取查询,其中至少在特定时间段内,每个调用中的结果都不会更改。

2. 缓存类型

通常,可以看到以下类型的缓存。

2.1. 内存中缓存

这是最常使用的区域,在该区域广泛使用缓存来提高应用程序的性能。 内存缓存(例如 Memcached 和 Radis)是应用程序和数据存储之间的键值存储。 由于数据保存在 RAM 中,因此它比将数据存储在磁盘上的典型数据库快得多。

RAM 比磁盘更受限制,因此高速缓存失效算法(例如最近最少使用(LRU))可以帮助使“冷”条目失效并将“热”数据保留在 RAM 中。 Memcached是一种内存缓存,其中Redis更高级,它允许我们备份和还原功能,它是分布式缓存工具,可以管理分布式集群中的缓存。

2.2. 数据库缓存

您的数据库通常在默认配置中包括某种程度的缓存,这些缓存针对通用用例进行了优化。 针对特定的使用模式调整这些设置可以进一步提高性能。 在该领域流行的是Hibernate或任何 ORM 框架的一级缓存。

2.3. Web 服务器缓存

反向代理和缓存(例如 Varnish)可以直接提供静态和动态内容。 Web 服务器还可以缓存请求,返回响应而无需联系应用程序服务器。 在当今的 API 时代,如果我们想在网络服务器级别缓存 API 响应,则此选项是可行的。

2.4. CDN 缓存

缓存可以位于客户端(操作系统或浏览器),服务器端或不同的缓存层中。

3. Spring Boot 缓存注解

Spring 框架为不同的缓存提供程序提供缓存抽象 api 。 API 的用法非常简单,但功能非常强大。 今天,我们将在缓存中看到基于注解的 Java 配置。 注意,我们也可以通过 XML 配置实现类似的功能。

3.1. @EnableCaching

它启用了 Spring 的注解驱动的缓存管理功能。 在 spring boot 项目中,我们需要将其添加到带有@SpringBootApplication注解的启动应用程序类中。 Spring 提供了一个并发哈希图作为默认缓存,但是我们可以重写CacheManager来轻松注册外部缓存提供程序。

3.2. @Cacheable

它在方法级别上用于使 spring 知道该方法的响应是可缓存的。 Spring 管理此方法对注解属性中指定的缓存的请求/响应。 例如,@Cacheable ("cache-name1", “cache-name2”)

@Cacheable注解具有更多选项。 就像我们可以从方法的请求中指定缓存的键一样。 如果未指定任何内容,spring 将使用所有类字段并将其用作缓存键(主要是HashCode)来维护缓存,但是我们可以通过提供键信息来覆盖此行为。

  1. @Cacheable(value="books", key="#isbn")
  2. public Book findStoryBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed)
  3. @Cacheable(value="books", key="#isbn.rawNumber")
  4. public Book findStoryBook (ISBN isbn, boolean checkWarehouse, boolean includeUsed)
  5. @Cacheable(value="books", key="T(classType).hash(#isbn)")
  6. public Book findStoryBook (ISBN isbn, boolean checkWarehouse, boolean includeUsed)

我们也可以使用条件缓存。 例如,

  1. @Cacheable(value="book", condition="#name.length < 50")
  2. public Book findStoryBook (String name)

3.3. @CachePut

有时我们需要手动操作缓存,以在方法调用之前放置(更新)缓存。 这将允许我们更新缓存,也将允许执行该方法。 该方法将始终执行,并将其结果放入缓存(根据@CachePut选项)。

它支持与@Cacheable相同的选项,应该用于缓存填充,而不是方法流程优化。

请注意,一般不建议在同一方法上使用@CachePut和`注解批注,因为它们的行为不同。 后者导致通过使用缓存跳过方法执行,而前者则强制执行以便执行缓存更新。

这会导致意外的行为,并且除了特定的极端情况(例如具有相互排斥条件的注解)外,应避免此类声明。

3.4. @CacheEvict

当我们需要移出(删除)先前加载的主数据的缓存时使用它。 当将执行CacheEvict注解的方法时,它将清除缓存。

我们可以在此处指定键以删除缓存,如果需要删除缓存的所有条目,则需要使用allEntries=true。 当需要清除整个缓存区域时,此选项非常有用 – 而不是逐出每个条目(由于效率低下,这将需要很长时间),所有条目都将在一次操作中被删除。

3.5. @Cache

当我们同时需要CachePutCacheEvict时,需要此注解。

4. 如何在 Spring Boot 中注册缓存引擎

Spring Boot 提供了与以下缓存提供程序的集成。 如果在类路径中存在默认选项,并且我们已通过 Spring Boot 应用程序中的@EnableCaching启用了缓存,则 Spring Boot 会使用默认选项进行自动配置。

  • JCache(JSR-107)(EhCache 3,Hazelcast,Infinispan 等)
  • EhCache 2.x
  • Hazelcast
  • Infinispan
  • Couchbase
  • RedisC
  • Caffeine
  • SimpleCache

我们可以通过覆盖特定于缓存提供程序的设置来覆盖 Spring 运行中的特定缓存行为,例如:

  1. spring.cache.infinispan.config=infinispan.xml

有关详细信息,我们可以在此处查看官方的 Spring Boot 文档

5. Spring Boot 缓存示例

在此 spring boot cahce 配置示例中,我们将看到如何在 spring boot 中启用默认缓存,并为一种业务方法启用缓存。 最后,我们将在重复调用相同方法的情况下测试应用程序性能。

我们将使用Thread.sleep()方法来模拟实际方法调用中的延迟,以感受缓存的效果。 因此,让我们遵循创建项目和测试的简单步骤。

5.1 创建 Spring Boot 项目

创建一个名为spring-cache且具有spring-boot-web依赖关系的简单 Spring Boot 项目,以将其托管在 Web 服务器中。

为此,我们需要转到 https://start.spring.io/ 并提供 Maven 坐标并选择依赖项。 下载包含框架项目的 zip 文件。 然后,一旦解压缩到合适的文件夹中,我们就需要将其导入 eclipse 中。 进行初始 mvn 全新安装,以将所有必需的依赖项下载到本地存储库。

Spring Boot 缓存示例教程 - 图1

Spring Boot 项目创建

5.2 创建 HTTP GET REST API

使用 GET 请求创建一个 REST 服务,它将成为搜索服务。 我们的主要目标是将方法的响应缓存在服务层中,在此我们将引入一个故意的延迟来模拟实际的后端服务调用以获取结果。 在第一个匹配中,响应将被延迟,因为我们在应用程序中会有一些模拟的延迟,但是在随后的调用中,我们将获得更快的响应。

Student.java

  1. package com.example.springcache.domain;
  2. public class Student {
  3. String id;
  4. String name;
  5. String clz;
  6. public Student(String id, String name, String clz) {
  7. super();
  8. this.id = id;
  9. this.name = name;
  10. this.clz = clz;
  11. }
  12. //Setters and getters
  13. }

StudentService.java

  1. package com.example.springcache.service;
  2. import org.springframework.cache.annotation.Cacheable;
  3. import org.springframework.stereotype.Service;
  4. import com.example.springcache.domain.Student;
  5. @Service
  6. public class StudentService
  7. {
  8. @Cacheable("student")
  9. public Student getStudentByID(String id)
  10. {
  11. try
  12. {
  13. System.out.println("Going to sleep for 5 Secs.. to simulate backend call.");
  14. Thread.sleep(1000*5);
  15. }
  16. catch (InterruptedException e)
  17. {
  18. e.printStackTrace();
  19. }
  20. return new Student(id,"Sajal" ,"V");
  21. }
  22. }

StudentController.java

  1. package com.example.springcache.controller;
  2. import org.springframework.beans.factory.annotation.Autowired;
  3. import org.springframework.web.bind.annotation.GetMapping;
  4. import org.springframework.web.bind.annotation.PathVariable;
  5. import org.springframework.web.bind.annotation.RestController;
  6. import com.example.springcache.domain.Student;
  7. import com.example.springcache.service.StudentService;
  8. @RestController
  9. public class StudentController
  10. {
  11. @Autowired
  12. StudentService studentService;
  13. @GetMapping("/student/{id}")
  14. public Student findStudentById(@PathVariable String id)
  15. {
  16. System.out.println("Searching by ID : " + id);
  17. return studentService.getStudentByID(id);
  18. }
  19. }

请注意:

  • 服务层方法使用@Cacheable("student")进行了注解,如上所述,此注解启用了该特定方法中的缓存,并且缓存名称为Student
  • getStudentByID()方法中,使用Thread.sleep(1000*5)有意延迟 5 秒。 这仅仅是为了了解响应是来自缓存还是真实的后端。

5.3 启用 Spring 托管缓存

为此,我们只需要在 Spring Boot 应用程序类中添加@EnableCaching注解。

SpringCacheApplication.java

  1. package com.example.springcache;
  2. import org.springframework.boot.SpringApplication;
  3. import org.springframework.boot.autoconfigure.SpringBootApplication;
  4. import org.springframework.cache.annotation.EnableCaching;
  5. @SpringBootApplication
  6. @EnableCaching
  7. public class SpringCacheApplication {
  8. public static void main(String[] args) {
  9. SpringApplication.run(SpringCacheApplication.class, args);
  10. }
  11. }

5.4 演示

现在我们可以测试 Spring 缓存的默认缓存行为。 我们已经添加了必需的配置,并且通过 Spring Boot 变得更加容易。

要进行测试,只需通过$ mvn clean install命令再次构建项目,然后从命令行 Java 命令运行应用程序,或者仅从 IDE 中运行SpringCacheApplication。 它将在localhost 8080端口中启动应用程序。

要测试,请转到 url

  1. http://localhost:8080/student/1

您将获得一个Student对象的JSON响应。 需要注意的是,第一次响应至少需要 5 秒钟,然后相同 URL 的后续响应会更快。 如果您难以理解差异,则可以更改服务等级中的延迟时间。

现在更改 URL 以通过http://localhost:8080/student/2获得学生 ID 2,您将再次遇到延迟,但是在随后的调用中,响应将从缓存提供。

这是我系统上关于此的最后几行日志。 当调用实际服务时,我将获得Going to sleep for 5 Secs.. to simulate backend call.日志,而在后续调用中,我未得到该日志,这意味着从缓存提供响应。

Console

  1. Searching by ID : 1
  2. Going to sleep for 5 Secs.. to simulate backend call.
  3. Searching by ID : 1
  4. Searching by ID : 1
  5. Searching by ID : 1
  6. Searching by ID : 1
  7. Searching by ID : 1
  8. Searching by ID : 2
  9. Going to sleep for 5 Secs.. to simulate backend call.
  10. Searching by ID : 2
  11. Searching by ID : 2

7. Spring Boot 缓存总结

最后,今天我们已经看到了 spring 框架在特定于应用程序缓存的缓存区域中提供了什么。 此外,我们还看到了 Spring 中存在哪些注解来支持这一点。

希望本教程对您有所帮助。 在本文中,我们使用了备用缓存提供程序,即后台的ConcurrentHashMap。 下一步将是配置其他受支持的缓存引擎,例如 RedisEhcache 等。

下载源码

学习愉快!

参考文献:

SpringBoot 缓存文档

Spring 5 缓存文档