Ehcache 的简单使用

article/2025/11/8 18:32:05

文章目录

  • Ehcache 的简单使用
    • 背景
    • 使用
      • 版本
      • 配置
        • 配置项
        • 编程式配置
        • XML 配置
        • 自定义监听器
      • 验证
        • 示例代码
      • 改进代码
    • 备注
      • 完整示例代码
      • 官方文档

Ehcache 的简单使用

背景

当一个JavaEE-Java Enterprise Edition应用想要对热数据(经常被访问,很少被修改的数据)进行缓存时,在遥远的年代,还没有Redis,开发者们想到的是直接利用JDK中的集合进行缓存。随之而来的问题是,JVM内存毕竟有限,如果热数据太多,过期策略、驱逐策略这些都需要开发者手动编写。那么Ehcache是主要解决这类问题的。同时,如果应该目前只是单机应用,那Ehcache就更适合了,不需要引入Redis从而导致增加系统复杂度,只需要引入一个体积很小的jar包即可,而且Ehcache为了缓存数据的存储,提供了多种方案,例如:直接使用JVM内存、使用堆外内存、使用硬盘来存储。使得开发者专注于业务开发,而无需过多担心脱离业务的数据缓存问题。

Ehcache在单体应用方面表现优秀,Eachce的开发者也同样想到了集群的处理方案。本文只关注单体应用的Ehcache的使用。至于集群方面的处理,个人认为是现在的集群处理一般使用Redis,Ehcache的集群方案使用者不多

使用

示例程序使用一个非常简单的Springboot项目来说明

版本

目前Ehcache的最新版本是3。由于3和之前的版本2发生了较大的变化,所以版本3和2之间是不兼容的。在Ehcache版本2中,还提供了ehcache-web,该功能是缓存整个web页面的响应,结合Filter实现,这个功能在遥远的JavaEE年代的单机应用上,用着确实舒服。不过,版本3中去掉了这个功能,Ehcache3的开发者觉得此功能太过于细化,偏离了Ehcache的方向

配置

Ehcache3提供了XML文件配置和编程式配置方式。无论是XML配置还是编程式配置方式,配置项都是一样的,所以先了解下Ehcache中的常见的配置项

配置项

  • cache alias - cache别名,一个应用可能有多个cache,每个cache需要一个名字
  • cache key type - 具体一个cache的key的类型
  • cache value type - cache key 对应的value 类型
  • cache expiry - 过期策略, Ehcache提供三种,永不过期、timeToLive、timeToIdle
  • cache resources - 资源配置,配置一个cache中最大的资源数,以及资源的位置,如上文说的堆、堆外、硬盘
  • cache listeners - 监听器,主要是监听cache项的某一种事件,例如:创建cache项、删除、过期、更新等

基本配置项就这些,和我们在远古时代,自己写缓存考虑到的几个点基本一致。

编程式配置

@Slf4j
@Configuration
public class EhcacheConfiguration {public static final String CACHE_NAME = "demo";/*** 过期策略* no expiry* timeToLive* timeToIdle-this means cache mappings will expire after a fixed duration following the time they were last accessed* https://www.ehcache.org/documentation/3.9/expiry.html** 存储位置选择:* 1.堆* 2.堆外-需要自己定义资源池* 3.磁盘* 4.集群* https://www.ehcache.org/documentation/3.9/tiering.html** 驱逐策略:* 官方对ehcache3的驱逐策略给的资料较少,而且提示,驱逐时会降低效率。网上查资料有的说,在ehcache看来,所有的缓存对象都是等价的* https://www.ehcache.org/documentation/3.9/eviction-advisor.html* @return org.ehcache.CacheManager*/@Beanpublic CacheManager cacheManager(CacheEventListener<Object, Object> cacheEventListener) {return initCacheManagerFromProgrammatic(cacheEventListener);}public CacheManager initCacheManagerFromProgrammatic(CacheEventListener<Object, Object> cacheEventListener) {return CacheManagerBuilder.newCacheManagerBuilder().withCache(CACHE_NAME,CacheConfigurationBuilder.newCacheConfigurationBuilder(Long.class, DataVO.class, ResourcePoolsBuilder.heap(2))// 过期策略只能选一种,存在多种,后面的覆盖前面的.withExpiry(ExpiryPolicyBuilder.timeToLiveExpiration(Duration.ofSeconds(30))).withExpiry(ExpiryPolicyBuilder.timeToIdleExpiration(Duration.ofMinutes(2))).withExpiry(ExpiryPolicy.NO_EXPIRY)// 配置监听器.withService(initCacheEventListenerConfigurationBuilder(cacheEventListener))).build(true);}/*** cache监听器* @param cacheEventListener* @return*/private CacheEventListenerConfigurationBuilder initCacheEventListenerConfigurationBuilder(CacheEventListener<Object, Object> cacheEventListener) {return  CacheEventListenerConfigurationBuilder.newEventListenerConfiguration(cacheEventListener, EventType.CREATED, EventType.EXPIRED, EventType.UPDATED, EventType.REMOVED).unordered().asynchronous();}
}

XML 配置

<?xml version="1.0" encoding="UTF-8"?>
<ehcache:configxmlns:xsi='http://www.w3.org/2001/XMLSchema-instance'xmlns:ehcache='http://www.ehcache.org/v3'xsi:schemaLocation="http://www.ehcache.org/v3 https://www.ehcache.org/schema/ehcache-core-3.9.xsd"><ehcache:cache alias="demo"><ehcache:key-type>java.lang.Long</ehcache:key-type><ehcache:value-type>com.example.ehcache.vo.DataVO</ehcache:value-type><ehcache:expiry><ehcache:tti unit="minutes">1</ehcache:tti></ehcache:expiry><ehcache:listeners><ehcache:listener><ehcache:class>com.example.ehcache.config.CacheEventLogListener</ehcache:class><ehcache:event-firing-mode>ASYNCHRONOUS</ehcache:event-firing-mode><ehcache:event-ordering-mode>UNORDERED</ehcache:event-ordering-mode><!--定义多个监听事件--><ehcache:events-to-fire-on>CREATED</ehcache:events-to-fire-on><ehcache:events-to-fire-on>UPDATED</ehcache:events-to-fire-on><ehcache:events-to-fire-on>REMOVED</ehcache:events-to-fire-on><ehcache:events-to-fire-on>EXPIRED</ehcache:events-to-fire-on></ehcache:listener></ehcache:listeners><ehcache:resources><ehcache:heap unit="entries">10</ehcache:heap></ehcache:resources></ehcache:cache>
</ehcache:config>

注意:这里的listeners标签一定要放在resource标签之前

自定义监听器

我们根据Ehcache的官方接口CacheEventListener,自定义一个监听器,其作用主要是通过log打印事件名字、key、value的值

@Component
@Slf4j
public class CacheEventLogListener implements CacheEventListener<Object, Object> {@Overridepublic void onEvent(CacheEvent<? extends Object, ? extends Object> cacheEvent) {log.info("cacheType is {}, key is {}, oldValue {}, newValue {}", cacheEvent.getType().toString(), cacheEvent.getKey(), cacheEvent.getOldValue(), cacheEvent.getNewValue());}
}

验证

示例代码

我们使用一个简单Controller来验证Ehcache的缓存和自定义监听器的功能

@RestController
@RequestMapping(path = "/data")
public class CommonDataController {@PostMappingpublic Long createDataVO(@RequestBody DataVO data) {Random random = new Random();Long result = random.nextLong();data.setId(result);Cache<Long, DataVO> cache = cacheManager.getCache(EhcacheConfiguration.CACHE_NAME, Long.class, DataVO.class);cache.put(result, data);return result;}@GetMapping(path = "/{id}")public DataVO getCacheData(@PathVariable Long id) {Cache<Long, DataVO> cache = cacheManager.getCache(EhcacheConfiguration.CACHE_NAME, Long.class, DataVO.class);DataVO result;result = cache.get(id);if (Objects.isNull(result)) {throw new RuntimeException("cache not exist");}return result;}
}

这里只列出部分代码,通过一个POST接口和GET接口验证下

curl -X POST -H 'Content-Type: application/json' http://localhost:18080/ehcache3/data

请求成功会返回新数据的ID - 8585661300356241871
在这里插入图片描述

根据ID请求

curl -X GET http://localhost:18080/ehcache3/data/8585661300356241871

在这里插入图片描述
等到我们配置的过期策略过期之后,再请求同一个ID,可以由上面第一张截图那样,自定义监听器把过期事件打印出来了

改进代码

我们简单的改进一下代码,使得当前应用支持XML配置和编程式配置,在开启特定属性下,使用XML配置,否则使用默认配置。

在application.yml中增加属性如下

ehcache:read-from-xml: true

EhcacheConfiguration修改后的整体配置如下

@Slf4j
@Configuration
public class EhcacheConfiguration {public static final String CACHE_NAME = "demo";@Value("${ehcache.read-from-xml}")private Boolean readFromXml;/*** 过期策略* no expiry* timeToLive* timeToIdle-this means cache mappings will expire after a fixed duration following the time they were last accessed* https://www.ehcache.org/documentation/3.9/expiry.html** 存储位置选择:* 1.堆* 2.堆外-需要自己定义资源池* 3.磁盘* 4.集群* https://www.ehcache.org/documentation/3.9/tiering.html** 驱逐策略:* 官方对ehcache3的驱逐策略给的资料较少,而且提示,驱逐时会降低效率。网上查资料有的说,在ehcache看来,所有的缓存对象都是等价的* https://www.ehcache.org/documentation/3.9/eviction-advisor.html* @return org.ehcache.CacheManager*/@Beanpublic CacheManager cacheManager(CacheEventListener<Object, Object> cacheEventListener) {CacheManager result;if (readFromXml) {result = initCacheManagerFromXml();}else {result = initCacheManagerFromProgrammatic(cacheEventListener);}return result;}private CacheManager initCacheManagerFromXml() {URL resource = getClass().getResource("/ehcache.xml");Objects.requireNonNull(resource);XmlConfiguration xmlConfiguration = new XmlConfiguration(resource);CacheManager result = CacheManagerBuilder.newCacheManager(xmlConfiguration);result.init();return result;}public CacheManager initCacheManagerFromProgrammatic(CacheEventListener<Object, Object> cacheEventListener) {return CacheManagerBuilder.newCacheManagerBuilder().withCache(CACHE_NAME,CacheConfigurationBuilder.newCacheConfigurationBuilder(Long.class, DataVO.class, ResourcePoolsBuilder.heap(2))// 过期策略只能选一种,存在多种,后面的覆盖前面的.withExpiry(ExpiryPolicyBuilder.timeToLiveExpiration(Duration.ofSeconds(30))).withExpiry(ExpiryPolicyBuilder.timeToIdleExpiration(Duration.ofMinutes(2))).withExpiry(ExpiryPolicy.NO_EXPIRY)// 配置监听器.withService(initCacheEventListenerConfigurationBuilder(cacheEventListener))).build(true);}/*** cache监听器* @param cacheEventListener* @return*/private CacheEventListenerConfigurationBuilder initCacheEventListenerConfigurationBuilder(CacheEventListener<Object, Object> cacheEventListener) {return  CacheEventListenerConfigurationBuilder.newEventListenerConfiguration(cacheEventListener, EventType.CREATED, EventType.EXPIRED, EventType.UPDATED, EventType.REMOVED).unordered().asynchronous();}
}

到此为止,Ehcache的简单使用就是这样

备注

完整示例代码

  • 示例代码仓库

官方文档

  • 官方文档,更多详细的配置说明都可以在官方文档中找到

http://chatgpt.dhexx.cn/article/Nxv2SFq9.shtml

相关文章

SpringBoot 缓存(EhCache 使用)

SpringBoot 缓存(EhCache 使用) 源文链接&#xff1a;http://blog.csdn.net/u011244202/article/details/55667868 SpringBoot 缓存(EhCache 2.x 篇) SpringBoot 缓存 在 Spring Boot中&#xff0c;通过EnableCaching注解自动化配置合适的缓存管理器&#xff08;CacheManager…

shiro框架04会话管理+缓存管理+Ehcache使用

目录 一、会话管理 1.基础组件 1.1 SessionManager 1.2 SessionListener 1.3 SessionDao 1.4 会话验证 1.5 案例 二、缓存管理 1、为什么要使用缓存 2、什么是ehcache 3、ehcache特点 4、ehcache入门 5、shiro与ehcache整合 1&#xff09;导入相关依赖&#xff0…

使用Ehcache的两种方式(代码、注解)

Ehcache,一个开源的缓存机制&#xff0c;在一些小型的项目中可以有效的担任缓存的角色&#xff0c;分担数据库压力此外&#xff0c;ehcache在使用上也是极为简单&#xff0c; 下面是简单介绍一下ehcahce的本地使用的两种方式&#xff1a; 1&#xff0c;使用代码编写的方式使用…

EhCache常用配置详解和持久化硬盘配置

一、EhCache常用配置 EhCache 给我们提供了丰富的配置来配置缓存的设置&#xff1b; 这里列出一些常见的配置项&#xff1a; cache元素的属性&#xff1a; name&#xff1a;缓存名称 maxElementsInMemory&#xff1a;内存中最大缓存对象数 maxElementsOnDisk&#xff…

EhCache初体验

一、简介 EhCache 是一个纯Java的进程内缓存框架&#xff0c;具有快速、精干等特点。Ehcache是一种广泛使用的开源Java分布式缓存。主要面向通用缓存,Java EE和轻量级容器。它具有内存和磁盘存储&#xff0c;缓存加载器,缓存扩展,缓存异常处理程序,一个gzip缓存servlet过滤器,支…

setw()使用方法

使用setw(n)之前&#xff0c;要使用头文件iomanip 使用方法: #include<iomanip> 1、setw&#xff08;int n&#xff09;只是对直接跟在<<后的输出数据起作用&#xff0c;而在之后的<<需要在之前再一次使用setw&#xff1b; &#xff08;Sets the number of…

c语言iomanip头文件的作用,iomanip头文件的作用

在c程序里面经常见到下面的头文件 #include io代表输入输出&#xff0c;manip是manipulator(操纵器)的缩写(在c上只能通过输入缩写才有效。) 作用(推荐学习&#xff1a;C语言视频教程) 主要是对cin,cout之类的一些操纵运算子&#xff0c;比如setfill,setw,setbase,setprecisio…

QT学习C++(6)

立方体的类设计 设计立方体类&#xff0c;求出立方体的面积(2ad2ac2bc)和体积(a*b*c)&#xff0c;分别用全局函数和成员函数判断两个立方体是否相等&#xff1f; #include <iostream>using namespace std; class Cube{ private://数据&#xff0c;长宽高int c_l;int c_w…

C++中使用setw()使用方法

setw(int n)是c中在输出操作中使用的字段宽度设置&#xff0c;设置输出的域宽&#xff0c;n表示字段宽度。只对紧接着的输出有效&#xff0c;紧接着的输出结束后又变回默认的域宽。当后面紧跟着的输出字段长度小于n的时候&#xff0c;在该字段前面用空格补齐&#xff1b;当输出…

关系代数表达式的优化

查询的处理的代价通常取决于磁盘访问&#xff0c;磁盘访问比内存访问速度慢很多。 在这里由于计算机原理的知识的欠缺&#xff0c;理解起来有点费劲&#xff0c;例如不知道关系的连接在哪里进行&#xff0c;连接的中间结果放在哪里&#xff0c;计算后的结果怎么处理&#xff0c…

关系代数1

转自链接&#xff1a; https://blog.csdn.net/Flora_SM/article/details/84190119 1.查询选修了2号课程的学生的学号。 2.查询至少选修了一门其直接先行课为5号课程的学生姓名 因为是选修直接先行课&#xff0c;所以在Course表里&#xff0c;而学生姓名在Student表里&#xff…

关系代数和SQL语法

数据分析的语言接口 OLAP计算引擎是一架机器&#xff0c;而操作这架机器的是编程语言。使用者通过特定语言告诉计算引擎&#xff0c;需要读取哪些数据、以及需要进行什么样的计算。编程语言有很多种&#xff0c;任何人都可以设计出一门编程语言&#xff0c;然后设计对应的编译…

关系代数表达式练习(针对难题)

教师关系T&#xff08;T#,TNAME,TITLE&#xff09;课程关系C(C#,CNAME,TNO)学生关系S(S#,SNAME,AGE,SEX)选课关系SC(S#,C#,SCORE) 检索至少选修了C2,C4两门课程的学生学号&#xff1a; 这里的下标可以这样理解&#xff0c;课程表C取了别名SC1,SC2,SC1的第一个元素&#xff08;…

怎样用关系代数表达式表示查询要求?求过程

怎样用关系代数表达式表示查询要求&#xff1f; 用一个例子来讲述一下 题目&#xff1a;查询至少选修了全部课程的学生学号和姓名 题目所用到的表如下 题目&#xff1a;查询至少选修了全部课程的学生学号和姓名&#xff1f; ① 找出题目中暗含属性、以及它们所在的表 ② 根据…

关系代数与sql语句

关系代数定义&#xff1a; 关系代数是以关系为运算对象的一组高级运算的集合。关系代数的运算有集合运算&#xff08;集合<表>与集合<表>之间的运算&#xff09;和关系运算&#xff08;集合<表>内部的运算&#xff09; 集合运算&#xff1a; 并运算&#xf…

关系代数2

转载链接&#xff1a; https://blog.csdn.net/Bruce_why/article/details/46389603 题A 设有如下所示的关系S(S#,SNAME,AGE,SEX)、C(C#,CNAME,TEACHER)和SC(S#,C#,GRADE)&#xff0c;用关系代数表达式表示下列查询语句&#xff1a; (1) 检索“程军”老师所授课程的课程号(C#)…

【数据库作业10】用SQL语句来表示关系代数中的表达式

1、有两个关系S&#xff08;A,B,C,D)和T&#xff08;C,D,E,F&#xff09;&#xff0c;写出与下列查询等价的SQL表达式&#xff1a; &#xff08;1&#xff09; σ A 10 ( S ) \sigma_{A10}(S) σA10​(S) //选择 select * from S where A10; &#xff08;2&#xff09; Π A…

① 数据库介绍 及 关系型数据库的关系代数表达式

查看专栏更多内容&#xff1a; ① 数据库介绍 及 关系型数据库的关系代数表达式 ② 关系数据库标准语言SQL 数据定义&#xff08;创建、修改基本表&#xff09;、数据更新&#xff08;增删改&#xff09; ③ 关系数据库标准语言SQL 数据查询&#xff08;SELECT&#xff09; …

关系代数表达式优化步骤

关系代数表达式优化步骤 本篇主要讲解怎么画查询语法树并对其优化&#xff0c;因为我在学关系代数的语法树的时候&#xff0c;在网上找不到比较详细的教法或者技巧&#xff0c;最后通过答案反推原理&#xff0c;所以想写一篇技巧来描述一下这类题的解题方法。 先上书内讲解 …

关系代数表达式学习

一、关系代数的9种操作&#xff1a; 关系代数中包括了&#xff1a;并、交、差、乘、选择、投影、联接、除、自然联接等操作。 五个基本操作&#xff1a; 并(∪)、差(-)、笛卡尔积()、投影(π)、选择(σ) 四个组合操作&#xff1a; 交(∩)、联接(等值联接)、自然联接(RS)、除法(…