在 Spring 中使用 @EhCache 注解作为缓存

文章目录

    • 项目概况
    • 项目设置
    • 一个简单的 RESTful Web 服务
      • Spring 整合 EhCache
      • 第 1 步:更新依赖项以使用 EhCache Spring 注解
      • 第 2 步:设置自定义缓存管理器
      • 第 3 步:配置 EhCache
      • 第 4 步:测试缓存
    • 刷新缓存
    • 总结
    • 推荐阅读文章

EhCache 是一种广泛使用的纯 Java 缓存,可以轻松地与大多数流行的 Java 框架集成,例如 SpringHibernate

它通常被认为是 Java 应用程序最方便的选择,因为它可以轻松集成到项目中。EhCache Spring Annotations 允许无缝集成到任何 Spring 应用程序中,只需向可缓存方法添加注释即可,而无需修改方法实现。本文重点介绍如何使用 EhCache Spring Annotations 提升您的 Spring 应用程序。


EhCache 是一种广泛使用的纯 Java 缓存,可以轻松地与大多数流行的 Java 框架集成,例如 SpringHibernate。它通常被认为是 Java 应用程序最方便的选择,因为它可以轻松集成到项目中。特别:

  • 只需将 JAR 包含在项目中即可。无需额外的安装步骤。
  • 与应用程序在相同的进程中运行,因此速度很快。无需其他服务即可运行。

简而言之,EhCache 是任何纯 Java 应用程序的绝佳选择。

在这里插入图片描述

虽然 EhCache 提供了简单、丰富的 API 来以编程方式操作缓存,但本文主要关注使用 EhCache Spring Annotations 以侵入性较小的方式提升 Spring 应用程序。我们将设置一个 Spring MVC 项目,并在 Tomcat 中部署一个 RESTful Web 服务。然后,EhCache 将集成到 Web 服务中。

项目概况

我们将在示例项目的上下文中演示 EhCache Annotations。我们将设置一个托管在 Tomcat 8 服务器上的基于 Spring MVC 的 Web 服务。

我在 Eclipse 中开发了该项目,可以按照Eclipse下载的说明进行安装。

当然,这些特定平台不是 EhCache 的要求;您始终可以选择自己喜欢的 IDE 和服务器。

EhCache Spring Annotations JAR 可Spring-EhCache下载获得。正如我们所看到的,每个版本都有两个 JAR:一个有依赖项,一个没有依赖项。具有依赖项的还包括 EhCache 2 和 Spring 3,它们是 EhCache 注解工作所必需的。如果我们下载带有依赖项的那个并将其添加到我们的构建路径中,则设置起来会更容易。

EhCache Spring Annotations 也与 Spring 4 兼容,但必须单独配置。目前尚不清楚该项目是否会在不久的将来支持 EhCache 3。对于正在使用或打算使用 EhCache 3 的用户,不建议使用本文中讨论的注释方法。

最后,我们将使用 Maven 来管理所有内容。Maven 预装在大多数 Eclipse 安装中,但也可以Maven官网获取。Spring MVCEhCache Spring Annotations 依赖项可以相当容易地添加,如本文后面所示。

项目设置

如果您以前从未设置过 Spring 项目,您可能还会发现 SpringMVC搭建过程提供了丰富的信息。

在本演示中,我们将使用 Maven官网 maven-archetype-webapp 设置一个基本项目。整体文件结构将如下所示:

在这里插入图片描述

创建一个目录 src/main/java,其中包含三个包:com.toptal.blogcom.toptal.blog.cachecom.toptal.blog.service。我们的应用程序源将进入这些包中,如下所述。

让我们在 web.xml 中定义一个名为 “springrest” 的 Tomcat servlet:

<web-app>...<servlet><servlet-name>springrest</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><load-on-startup>1</load-on-startup></servlet><servlet-mapping><servlet-name>springrest</servlet-name><url-pattern>/*</url-pattern></servlet-mapping>
</web-app>

除非另有明确说明,否则 Spring MVC DispatcherServlet 将在目录 WEB-INF 中查找名为 {servlet-name}-servlet.xml 的 XML 配置文件。让我们创建一个名为 springrest-servlet.xml 的配置文件。要启用带有 @RequestMapping 注释的 Spring 进程控制器方法,我们只需将 <mvc:annotation-driven /> 添加到此文件中即可。此外,让我们定义 Spring 的基本包,以便通过添加 <context:component-scan base-package="com.toptal.blog" /> .springrest-servlet.xml 配置将变为:

<beans ... ><mvc:annotation-driven /><context:component-scan base-package="com.toptal.blog" />
</beans>

一个简单的 RESTful Web 服务

现在,我们的项目已正确配置,让我们实现一个简单的 “消息服务” API。在我们的基础包 project.toptal.blog 中,我们将添加 SpringRestControllerWithEhCache.java ,其中包含一个按 ID 获取消息的 GET 方法,以及一个按 ID 设置消息的 POST 方法:

@RestController  
@RequestMapping( "/" )
public class SpringRestControllerWithEhCache {@AutowiredMessageService messageService;@RequestMapping( value = "/message/{id}", method = RequestMethod.GET )public String getMessage( @PathVariable Integer id ) {String message = messageService.getMessage( id );System.out.println( "get message ["+message+"] at "+new Date() );return message;}@RequestMapping( value = "/message/set/{id}/{message}", method = RequestMethod.POST )public String setMessage( @PathVariable Integer id, @PathVariable String message ) { System.out.println( "set message ["+message+"] at "+new Date() );messageService.setMessage( id, message );return message;  }
}

我们将在 com.toptal.blog.service 中定义 MessageService 类。它将访问存储在我们的记录系统 (SOR) 中的消息。在生产应用程序中,SOR 类似于关系数据库。为简单起见,我们将使用 HashMap

@Service
public class MessageService {private ConcurrentHashMap<Integer, String> messages= new ConcurrentHashMap<Integer, String>();public String getMessage( Integer id ) {System.out.println( "Getting data from SOR......" );return messages.get( id );}public void setMessage( Integer id, String message ){messages.put( id, message );}
}

现在,如果我们将项目导出为 WAR 并将其部署到 Tomcat 中,我们应该能够通过在 中创建 HTTP POST 请求来为 ID=1 设置一条消息,例如“test_message http://localhost:8080/EhCacheExample/message/set/1/test_message ”。然后,我们应该能够通过 HTTP GET 请求的 “test_message” 返回 http://localhost:8080/EhCacheExample/message/1 。我使用 Insomnia 作为方便的 REST 客户端来进行测试。

Spring 整合 EhCache

现在让我们让 EhCache 为我们工作。只需几个快速步骤即可配置我们的项目以正确运行 EhCache。

在这里插入图片描述

第 1 步:更新依赖项以使用 EhCache Spring 注解

在 Maven 的 pom.xml中添加 EhCache Spring Annotations 依赖项:

<!-- ehcache -->
<dependency><groupId>com.googlecode.ehcache-spring-annotations</groupId><artifactId>ehcache-spring-annotations</artifactId><version>1.2.0</version>
</dependency>

第 2 步:设置自定义缓存管理器

Spring 有一个内置的 EhCache 缓存管理器 org.springframework.cache.ehcache.EhCacheManagerFactoryBean 。这适用于大多数缓存情况,但我发现定义自定义缓存管理器非常有用,因为它允许我使用相同的缓存管理器以编程方式或通过注释控制缓存。本文重点介绍 annotations,但让我们继续定义一个自定义缓存管理器,以便在需要时做好准备。如果您更喜欢坚持使用 默认缓存管理器 ,您可以跳过此步骤。

我们将在 : com.toptal.blog.cache.CustomCacheManager

public class CustomCacheManager extends net.sf.ehcache.CacheManager{public CustomCacheManager(){super();}/* Add your own cache methods here.* * public void myCustomCacheMethod(){*    // your code here* }* */
}

通过更新springrest-servlet.xml来启用它,如下所示:

   ...<ehcache:annotation-driven cache-manager="customCacheManager" /><bean id="customCacheManager"class="com.toptal.blog.cache.CustomCacheManager"scope="singleton"></bean>...

第 3 步:配置 EhCache

最后,在 Classpath 中创建 EhCache 配置文件ehcache.xml。默认情况下,Eclipse 将在 classpath 中包含 src/main/resources,我们将文件放在这里。此文件是 EhCache 正常运行所必需的。它定义缓存名称和每个缓存的一些属性,例如 timeToLiveSeconds

<ehcache xmlms:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="ehcache.xsd"><diskStore path="cache" /><cachename="messageCache"maxElementsInMemory="10000"eternal="false"timeToIdleSeconds="0"timeToLiveSeconds="10"overflowToDisk="false"memoryStoreEvictionPolicy="LFU" />      
</ehcache>

第 4 步:测试缓存

现在,一切都设置好了,使用 EhCache 应该是一件简单而愉快的工作。我们可以简单地将 @Cacheable 添加到我们想要缓存的方法或类中。例如,我将 @Cacheable 添加到 MessageService 中的 getMessage 方法中。就是这么简单!

@Cacheable( cacheName = "messageCache" )
public String getMessage( Integer id ) {System.out.println( "Getting data from SOR......" );return messages.get( id );
}

要测试我们的缓存是否正常工作,我们可以通过在 处 http://localhost:8080/EhCacheExample/message/set/1/newMessage 发出 HTTP POST 请求来创建一条 ID=1 的消息,然后多次获取 ID=1 的消息,并向 . http://localhost:8080/EhCacheExample/message/1 如下面的控制台输出所示,Web 服务在我们第一次请求消息时要求 SOR 获取消息,但在接下来的两个请求中不请求,而是返回缓存的消息。由于我们将 timeToLiveSeconds 定义为 10,因此 Web 服务会在 10 秒后调用 SOR 再次获取消息:

set message [newMessage] at Sun Dec 06 23:55:39 MST 2015
get message [newMessage] at Sun Dec 06 23:55:42 MST 2015
Getting data from SOR......
get message [newMessage] at Sun Dec 06 23:55:47 MST 2015
get message [newMessage] at Sun Dec 06 23:55:49 MST 2015
get message [newMessage] at Sun Dec 06 23:55:54 MST 2015
Getting data from SOR......

刷新缓存

现在,我们正在享受缓存给我们带来的速度和便利,而且 EhCache 足够好,每 10 秒自行刷新一次。但是,如果我们想在 SOR 更新后立即刷新它,该怎么办?EhCache Spring Annotation 提供了@TriggersRemove,以便在调用带注释的方法时从缓存中删除指定的键。在我们的消息服务 API 中,当调用 setMessage 时,应该从缓存中删除缓存的消息。因此,下次收到 getMessage 请求时,缓存将从 SOR 中获取新记录:

@Cacheable(cacheName = "messageCache",keyGenerator = @KeyGenerator (                             // method name is not included in cache key to work with @TriggersRemovename = "HashCodeCacheKeyGenerator",properties = @Property( name="includeMethod", value="false" )))  
public String getMessage( Integer id ) {System.out.println( "Getting data from SOR......" );return messages.get( id );
}@TriggersRemove(cacheName = "messageCache",keyGenerator = @KeyGenerator (name = "HashCodeCacheKeyGenerator",properties = @Property( name="includeMethod", value="false" )))
public void setMessage( @PartialCacheKey Integer id, String message ) {messages.put( id, message );
}

缓存管理器使用密钥生成器来生成缓存密钥。可在此处找到预定义的缓存密钥生成器列表。默认情况下,@KeyGenerator 使用方法名称和传入的参数来生成缓存键。但是,由于我们希望 setMessage 方法生成与 getMessage 相同的 key,并删除与该 key 关联的缓存值,因此我们必须仅使用消息 ID 作为 key,并消除生成 key 的方法名称。因此,我们将这两个方法的密钥生成器的 includeMethod 属性设置为 false。此外,由于 setMessage 有两个参数,我们在 id 参数上使用 EhCache 的 @PartialCacheKey 注解来指定它是密钥生成器唯一应该使用的参数。最后,回想一下,我们为此资源类型配置了专用缓存 messageCache,因此仅使用键的 ID 不会与其他资源类型发生冲突。

现在,如果我们对 ID=1 的消息执行多个 HTTP 请求,如下所示:

HTTP POST:  http://localhost:8080/EhCacheExample/message/set/1/newMessage1
HTTP GET:http://localhost:8080/EhCacheExample/message/1
HTTP POST: http://localhost:8080/EhCacheExample/message/set/1/newMessage2
HTTP GET:http://localhost:8080/EhCacheExample/message/1

控制台将显示:

set message [newMessage1] at Tue Dec 08 17:53:44 MST 2015
get message [newMessage1] at Tue Dec 08 17:53:47 MST 2015
Getting data from SOR......
set message [newMessage2] at Tue Dec 08 17:53:50 MST 2015
get message [newMessage2] at Tue Dec 08 17:53:53 MST 2015
Getting data from SOR......

总结

最终的项目结构如下所示:

在这里插入图片描述

在此示例中,我们首先创建了一个简单的 Spring MVC RESTful Web 应用程序。无需修改现有应用程序代码的哪怕一行,我们就可以使用 EhCache Spring Annotations 将 EhCache 无缝集成到应用程序中。我们已经证明 EhCache Spring Annotations 既易于安装(通过添加其 Maven 依赖项)又易于使用(通过向方法添加注释)。

推荐阅读文章

1、使用 Spring 框架构建 MVC 应用程序:初学者教程
2、有缺陷的 Java 代码:Java 开发人员最常犯的 10 大错误
3、如何理解应用 Java 多线程与并发编程?
4、Java Spring 中常用的 @PostConstruct 注解使用总结
5、线程 vs 虚拟线程:深入理解及区别
6、深度解读 JDK 8、JDK 11、JDK 17 和 JDK 21 的区别
7、10大程序员提升代码优雅度的必杀技,瞬间让你成为团队宠儿!
8、“打破重复代码的魔咒:使用 Function 接口在 Java 8 中实现优雅重构!”
9、Java 中消除 If-else 技巧总结
10、线程池的核心参数配置(仅供参考)
11【人工智能】聊聊Transformer,深度学习的一股清流(13)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/447908.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Visual Studio的实用调试技巧总结

对于很多学习编程的老铁们来说&#xff0c;是不是也像下面这张图一样写代码呢&#xff1f; 那当我们这样编写代码的时候遇到了问题&#xff1f;大家又是怎么排查问题的呢&#xff1f;是不是也像下面这张图一样&#xff0c;毫无目的的一遍遍尝试呢&#xff1f; 这篇文章我就以 V…

k8s的微服务

ipvs模式 Service 是由 kube-proxy 组件&#xff0c;加上 iptables 来共同实现的 kube-proxy 通过 iptables 处理 Service 的过程&#xff0c;需要在宿主机上设置相当多的 iptables 规则&#xff0c;如果宿主机有大量的Pod&#xff0c;不断刷新iptables规则&#xff0c;会消耗…

QT:计算点到线段的垂线段的距离

描述 在Qt中&#xff0c;要计算一个点到一条线段的垂线段的长度&#xff08;即点到线段上最近点的距离&#xff0c;且这个点是垂直于线段的&#xff09;&#xff0c;你不能直接使用QVector2D::distanceToLine&#xff0c;因为这个方法计算的是点到直线的垂直距离&#xff0c;而…

2024-09-22 进程优先级,进程切换

一、僵尸状态 & 孤儿进程 进程退出&#xff1a;内核数据结构&#xff08;task_struct 维护&#xff09; 代码和数据&#xff08;直接释放&#xff09; 代码不会执行了首先可以立即释放的就是进程对应的程序信息数据。进程退出要有退出信息&#xff08;进程的退出码&#…

六西格玛设计DFSS方法论在消费级无人机设计中的应用——张驰咨询

本文基于六西格玛设计方法论&#xff0c;对消费级无人机的设计流程进行系统化研究&#xff0c;探讨如何通过六西格玛设计的理念、工具和方法提升无人机产品的设计质量和市场竞争力。文章从市场定位、客户需求分析出发&#xff0c;深入到关键KPI指标的制定&#xff0c;并逐步阐述…

【网络安全】漏洞案例:提升 Self-XSS 危害

未经许可,不得转载。 文章目录 Self-XSS-1Self-XSS-2Self-XSS-1 目标应用程序为某在线商店,在其注册页面的First Name字段中注入XSS Payload: 注册成功,但当我尝试登录我的帐户时,我得到了403 Forbidden,即无法登录我的帐户。 我很好奇为什么我无法登录我的帐户,所以我…

前端高频面试题2024/9/22(偏项目问题--通用后台管理系统)

文章目录 一.前端项目概述1.系统登录注册模块1.对注册的密码进行加密 &#xff08;使用加密中间件bcrypt.js&#xff09;2.登录成功后返回token3.前端登录页面有用到弹性布局&#xff0c;ref和reactive4.登录头像&#xff1a;文件上传 2.系统设置模块2.系统首页模块&#xff08…

使用electron+vue开发桌面应用

electron是什么 electron由Node.jsChromiumNative APIs构成。你可以理解成&#xff0c;它是一个得到了Node.js和基于不同平台的Native APIs加强的Chromium浏览器&#xff0c;可以用来开发跨平台的桌面级应用。 它的开发主要涉及到两个进程的协作——Main&#xff08;主&#x…

Java通过RAG构建专属知识问答机器人_超详细

RAG&#xff1a;融合检索与生成的文本精准生成技术 检索增强生成&#xff08;RAG&#xff09;是一种技术&#xff0c;它通过结合检索模型和生成模型来提高文本生成的准确性。具体来说&#xff0c;RAG首先利用检索模型从私有或专有的数据源中搜索相关信息&#xff0c;然后将这些…

编程练习7 5G网络建设

需要用到并查集的相关知识&#xff1a;可以参考如下链接 并查集详解&#xff08;原理代码实现应用优化&#xff09;-CSDN博客 #include<iostream> #include<algorithm> #include<vector>using namespace std;vector<int> split(string params_str) {…

ICT产业新征程:深度融合与高质量发展

在信息时代的浪潮中&#xff0c;每一场关于技术革新与产业融合的盛会都闪耀着智慧的光芒&#xff0c;引领着未来的方向。9月25日&#xff0c;北京国家会议中心内&#xff0c;一场聚焦全球信息通信业的顶级盛事——第32届“国际信息通信展”&#xff08;PT展&#xff09;隆重拉开…

【LwIP源码学习3】TCP协议栈分析——数据接收流程

前言 本文介绍代码在lwip的tcp_in.c文件中&#xff0c;主要介绍TCP协议栈中数据的接收流程。 正文 1、一个正常的TCP数据&#xff0c;首先会传入到 tcp_input(struct pbuf *p, struct netif *inp)函数&#xff0c;其中指针p指向传入的数据流。 2、从数据流中获取TCP头部 …

通过Express + Vue3从零构建一个用户认证与授权系统(二)数据库与后端项目搭建与实现

前言 上一篇完成了系统的相关设计文档的编写&#xff0c;本文将详细介绍如何一步步使用 TypeScript 和 Express 搭建一个模块化、类型安全的用户认证与授权系统&#xff0c;包括数据库设计、后端项目搭建、用户认证、角色与权限管理、错误处理以及 Swagger 文档集成。 项目准…

【ubuntu】ubuntu20.04安装cuda12.6与显卡驱动

目录 1.安装cuda12.6 2.安装显卡驱动 1.安装cuda12.6 https://developer.nvidia.com/cuda-toolkit-archive https://developer.nvidia.com/cuda-12-6-0-download-archive?target_osLinux&target_archx86_64&DistributionUbuntu&target_version20.04&target_…

DART: Implicit Doppler Tomography for Radar Novel View Synthesis 笔记

Link&#xff1a;https://wiselabcmu.github.io/dart/ Publish&#xff1a; 2024CVPR Abstract DART主要任务就是用来合成雷达距离多普勒图像range-droppler&#xff0c;可用于生成高质量的断层扫描图像。 Related Work 1 Radar Simulation 基于模型的方法 任务&#xff…

XGBoost回归预测 | MATLAB实现XGBoost极限梯度提升树多输入单输出

回归预测 | MATLAB实现XGBoost极限梯度提升树多输入单输出 目录 回归预测 | MATLAB实现XGBoost极限梯度提升树多输入单输出预测效果基本介绍模型描述程序设计参考资料预测效果 基本介绍 XGBoost的全称是eXtreme Gradient Boosting,它是经过优化的分布式梯度提升库,旨在高效、…

医学和生信web APP 平台- Appmatrix

医学&#xff08;和生信&#xff09;web APP 平台- Appmatrix 最近使用shinyproxy将平时所构建的shiny和streamlit医学类应用汇集在一起&#xff0c;实现一站式访问&#xff0c;另外&#xff0c;使用了自己电脑内网穿透&#xff0c;一定程度上缓解了数据分析类APP消耗计算资源…

关闭线程池的API介绍

线程池关闭主要涉及以下几个API: shutdown:关闭线程池&#xff0c;此方法执行后&#xff0c;线程池不会立即关闭&#xff0c;1、等待正在执行的线程任务执行完毕 2、等待任务队列中的任务执行完毕&#xff0c;步骤1和2执行后&#xff0c;线程池才完全中止isShutdown:判断线程池…

PostgreSQL学习笔记六:模式SCHEMA

模式&#xff08;Schema&#xff09; PostgreSQL中的模式&#xff08;Schema&#xff09;是一个命名的数据库对象集合&#xff0c;包括表、视图、索引、数据类型、函数、存储过程和操作符等。模式的主要作用是组织和命名空间数据库对象&#xff0c;使得同一个数据库中可以包含…

C/C++语言基础--C++异常看这一篇就够了

本专栏目的 更新C/C的基础语法&#xff0c;包括C的一些新特性 前言 通过前面几节课&#xff0c;我们学习了抽象、封装、继承、多态等相关的概念&#xff0c;接下来我们将讲解异常&#xff0c;异常是专门处理错误的&#xff1b;这一次加了不少图标&#xff0c;希望大家喜欢;C语…