缓存之美 —— 如何选择合适的本地缓存?
创始人
2024-05-16 13:45:37
0

原标题:缓存之美 —— 如何选择合适的本地缓存?

作者:京东云开发者-京东保险 郭盼

链接:https://my.oschina.net/u/4090830/blog/10768446

1、简介

小编最近在使用系统的时候,发现尽管应用已经使用了 redis 缓存提高查询效率,但是仍然有进一步优化的空间,于是想到了比分布式缓存性能更好的本地缓存,因此对领域内常用的本地缓存进行了一番调研,有早期的 Guava 缓存、在 Guava 上进一步传承的 Caffine 以及自称在 Java 中使用最广泛的 EhCache,那么我们该怎么选择适合自己应用的缓存呢,小编下面会简单介绍,并将以上缓存进行一个对比,希望帮助大家选择最适合自己系统的本地缓存。

2、Guava 缓存简介

Guava cache 是 Google 开发的 Guava 工具包中一套完善的 JVM 本地缓存框架,底层实现的数据结构类似于 ConcurrentHashMap,但是进行了更多的能力拓展,包括缓存过期时间设置、缓存容量设置、多种淘汰策略、缓存监控等,下面简单介绍下这些功能及其使用方式。

2.1、缓存过期时间设置

Guava 的过期时间设置有基于创建时间和最后一次访问时间两种策略.

(1) 基于创建时间

通过对比缓存记录的插入时间来判断,比如设置过期时间为 5 分钟,不管中间有没有访问,到时过期。

publicCache<String, String> createCache {

returnCacheBuilder.newBuilder

.expireAfterWrite(5L, TimeUnit.MINUTES)

.build;

}

(2) 基于过期时间

通过对比最近最后一次的访问时间,比如设置 5 分钟,每次访问之后都会刷新过期时间为 5 分钟,只有持续 5 分钟没有被访问到才会过期。

publicCache<String, String> createCache {

returnCacheBuilder.newBuilder

.expireAfterAccess(5L, TimeUnit.MINUTES)

.build;

}

2.2、缓存容量和淘汰策略设置

Guava cache 是内存型缓存,有内存溢出风险,因此需要设置缓存的最大存储上限,通过缓存的条数或每条缓存的权重来判断是否达到了设定阈值,当缓存的数据量达到设定阈值之后,Guava cache 支持使用 FIFO 和 LRU 的策略对缓存记录采取淘汰的措施。

(1)限制缓存记录条数

publicCache createCache{

returnCacheBuilder.newBuilder

.maximumSize(100L)

.build;

}

(2)限制缓存记录权重

publicCache createCache{

returnCacheBuilder.newBuilder

.maximumWeight(100L)

.weigher((key, value) -> (int) Math.ceil(instrumentation.getObjectSize(value) / 1024L))

.build;

}

使用限制缓存记录权重时要先计算 weight 的 value 对象的字节数,每 1kb 字节作为一个权重,对比限制缓存记录,我们就能将缓存的总占用限制在 100kb 左右。

2.3 缓存监控

缓存记录的加载和命中情况是评价缓存处理能力的重要指标,Guava cache 提供了 stat 统计日志对这两个指标进行了统计,我们只需要在创建缓存容器的时候加上 recordStats 就可以开启统计。

publicCache createCache{

returnCacheBuilder.newBuilder

.recordStats

.build;

}

2.4 Guava cache 的优劣势和适用场景

优劣势:Guava cache 通过内存处理数据,具有减少 IO 请求,读写性能快的优势,但是受内存容量限制,只能处理少量数据的读写,还有可能对本机内存造成压力,并且在分布式部署中,会存在不同机器节点数据不一致的情况,即缓存漂移。

适用场景:读多写少,对数据一致性要求不高的场景。

3、Caffeine 简介

Caffeine 同样是 Google 开发的,是在 Guava cache 的基础上改良而来的,底层设计思路、功能和使用方式与 Guava 非常类似,但是各方面的性能都要远远超过前者,可以看做是 Guava cache 的升级版,因此,之前使用过 Guava cache,也能够很快的上手 Caffeine,下面是 Caffeine 和 Guava cache 的缓存创建对比,基本可以无门槛过渡。

publicCache<String, String> createCache {

returnCaffeine.newBuilder

.initialCapacity(1000)

.maximumSize(100L)

.expireAfterWrite(5L, TimeUnit.MINUTES)

.recordStats

.build;

}

publicCache<String, String> createCache {

returnCacheBuilder.newBuilder

.initialCapacity(1000)

.maximumSize(100L)

.expireAfterWrite(5L, TimeUnit.MINUTES)

.recordStats

.build;

}

那么 Caffeine 底层又做了哪些优化,才能让其性能高于 Guava cache 呢?主要包含以下三点:

3.1、对比 Guava cache 的性能主要优化项

(1)异步策略

Guava cache 在读操作中可能会触发淘汰数据的清理操作,虽然自身也做了一些优化来减少读的时候的清理操作,但是一旦触发,就会降低查询效率,对缓存性能产生影响。而在 Caffeine 支持异步操作,采用异步处理的策略,查询请求在触发淘汰数据的清理操作后,会将清理数据的任务添加到独立的线程池中进行异步操作,不会阻塞查询请求,提高了查询性能。

(2)ConcurrentHashMap 优化

Caffeine 底层都是通过 ConcurrentHashMap 来进行数据的存储,因此随着 Java8 中对 ConcurrentHashMap 的调整,数组 + 链表的结构升级为数组 + 链表 + 红黑树的结构以及分段锁升级为 syschronized+CAS,降低了锁的粒度,减少了锁的竞争,这两个优化显著提高了 Caffeine 在读多写少场景下的查询性能。

(3)新型淘汰算法 W-TinyLFU

传统的淘汰算法,如 LRU、LFU、FIFO,在实际的缓存场景中都存在一些弊端,如 FIFO 算法,如果缓存使用的频率较高,那么缓存数据会一直处在进进出出的状态,间接影响到缓存命中率。LRU 算法,在批量刷新缓存数据的场景下,可能会将其他缓存数据淘汰掉,从而带来缓存击穿的风险。LFU 算法,需要保存缓存记录的访问次数,带来内存空间的损耗。

因此,Caffeine 引入了 W-TinyLFU 算法,由窗口缓存、过滤器、主缓存组成。缓存数据刚进入时会停留在窗口缓存中,这个部分只占总缓存的 1%,当被挤出窗口缓存时,会在过滤器汇总和主缓存中淘汰的数据进行比较,如果频率更高,则进入主缓存,否则就被淘汰,主缓存被分为淘汰段和保护段,两段都是 LRU 算法,第一次被访问的元素会进入淘汰段,第二次被访问会进入保护段,保护段中被淘汰的元素会进入淘汰段,这种算法实现了高命中率和低内存占用。更详细的解释可以参考论文:https://arxiv.org/pdf/1512.00727.pdf

3.2、Caffeine 的优劣势和适用场景

优势:对比 Guava cache 有更高的缓存性能,劣势:仍然存在缓存漂移的问题;JDK 版本过低无法使用

适用场景:1、适用场景:读多写少,对数据一致性要求不高的场景;2、纯内存缓存,JDK8 及更高版本中,追求比 Guava cache 更高的性能。

4、Ehcache 简介

Guava cache 和 Caffeine 都是 JVM 缓存,会受到内存大小的制约,最新的 Ehcache 采用堆内缓存 + 堆外缓存 + 磁盘的方式,打破了这一制约。堆内缓存就是被 JVM 管理的那一部分缓存,而堆外缓存,就是在内存中另外在开辟一块不被 JVM 管理的部分。堆外缓存这部分既可以享受内存的高速读写能力,而且又避免的 JVM 频繁的 GC,缺点是需要自行清理数据。

下面是 Ehcache 缓存的创建,指定了堆内、堆外缓存和磁盘缓存的大小。

ResourcePoolsBuilder.newResourcePoolsBuilder

.heap(20, MemoryUnit.MB)

.offheap(10, MemoryUnit.MB)

.disk(5, MemoryUnit.GB);

为了解决缓存漂移的问题,Ehcache 支持通过集群的方式,实现了分布式节点之间的数据互通。关于 Ehcache 的集群策略,后续文章再详细阐述。

5、不同本地缓存对比

框架 命中率 速度 回收算法 使用难度 集群 适用场景
Guava cache 第三 LRU、LFU、FIFO 不支持 读多写少,允许少量缓存偏移
Caffeine 第一 W-TinyLFU 不支持 读多写少,允许少量缓存偏移,能用 Caffeine 就别用 Guava cache
Ehcache 第二 LRU、LFU、FIFO 支持 分布式系统中对数据一致性要求高

END

AI 领域大佬欢聚一堂 跳起科目三

这里有最新开源资讯、软件更新、技术干货等内容

点这里 ↓↓↓ 记得 关注✔ 标星⭐ 哦~

相关内容

原创 ...
三大运营商的承诺该不该相信?希望能够相信,但似乎并不是那么令人信服...
2025-07-23 11:21:25
高端前端培训,如何让你从小...
最近总有人问我:"现在转行学前端还来得及吗?"我的回答永远是:当你...
2025-07-23 10:02:29
原创 ...
分析:凯文·杜兰特的来访如何帮助泰瑞斯·哈利伯顿的跟腱康复 泰瑞斯...
2025-07-23 05:02:07
原创 ...
前不久,美军7 架 B-2隐形轰炸机对伊朗实施“午夜之锤”军事行动...
2025-07-20 14:21:26
原创 ...
尽管俄军苏-57号称五代机,但由于其隐身能力不佳,被很多网友称为准...
2025-07-20 12:21:38
大厂为何正扎堆卷赛博“大白...
出品 | 虎嗅科技医疗组 作者 | 陈广晶 编辑 | 苗正卿 头图...
2025-07-20 08:21:23

热门资讯

存款利息怎么算?湖南农商行存款... 导读湖南农商行存款利息怎么算?湖南农商行作为湖南最大的股份制银行,它是一家地方性、集约化、国际化、股...
处暑节气如何做好养生?这些秘笈... 原标题:处暑节气如何做好养生?这些秘笈请收好 处暑的天气多变,可能给我们的身体...
原创 婴... 在宝宝的成长过程中,奶粉作为重要的营养来源,其选择至关重要。那么,什么样的婴儿奶粉比较好呢?奶粉的营...
原创 如... 肺是人体非常重要的器官,空气中的很多灰尘和垃圾都会随空气进入肺部,所以肺部的清洁是非常重要的。我们可...
金三角李国辉,率领三千残部打败... 原标题:金三角李国辉,率领三千残部打败泰国政府军,到台湾后结局如何? 解放战争...
新农保一年缴费180元60岁后... 导读新农保一年缴费180元,60岁后能拿多少钱呢?我们这一个月是208块。农村大多是买这种农保吧,刚...
孕妈检测出胎儿没有“胎心”究竟... 原标题:孕妈检测出胎儿没有“胎心”究竟为何?多半是这4种原因,预防下 昨天小樱...
激光术后护理指南:如何科学恢复... 二氧化碳激光是一种常用于去除瘢痕、色素沉着、皱纹和其他皮肤问题的治疗方法。它通过发射高能量的二氧化碳...
立春节气如何养生? 原标题:立春节气如何养生? 今天是2024年2月24日,阴历腊月二十五,今日立...
郑州银行大额存单利率:1、活期... 导读2022郑州银行大额存单利率一、存款利率:1、活期存款:0.3%2、整存整取:三个月1.60%,...