Java简单缓存实现方法解析
时间:2026-01-23 13:02:34 282浏览 收藏
小伙伴们对文章编程感兴趣吗?是否正在学习相关知识点?如果是,那么本文《Java实现简单缓存机制详解》,就很适合你,本篇文章讲解的知识点主要包括。在之后的文章中也会多多分享相关知识点,希望对大家的知识积累有所帮助!
ConcurrentHashMap 可实现线程安全简易缓存,支持原子操作如 computeIfAbsent,适合静态数据;需手动管理过期与清理,不自动回收;Weak/SoftReference 不适用因不可控且非线程安全;LinkedHashMap 非线程安全且无 TTL;复杂场景应选 Caffeine。

用 ConcurrentHashMap 实现线程安全的简易缓存
Java 标准库没有开箱即用的“轻量缓存类”,但 ConcurrentHashMap 足以支撑大多数简单场景——它支持高并发读写,且自带 computeIfAbsent 这类原子操作,避免手动加锁。
- 适合缓存不常变更、生命周期由业务控制(比如配置项、枚举映射表)的场景
- 不自动过期,需自行在写入/读取时检查时间戳或版本号
- 若键值对数量增长不可控,必须配合
size()+ 定期清理逻辑,否则可能内存泄漏
private final ConcurrentHashMap<String, CacheEntry> cache = new ConcurrentHashMap<>();
static class CacheEntry {
final Object value;
final long expireAt; // 毫秒时间戳
CacheEntry(Object value, long ttlMs) {
this.value = value;
this.expireAt = System.currentTimeMillis() + ttlMs;
}
}
public <T> T get(String key, Supplier<T> loader) {
CacheEntry entry = cache.get(key);
if (entry != null && System.currentTimeMillis() < entry.expireAt) {
return (T) entry.value;
}
T loaded = loader.get();
cache.put(key, new CacheEntry(loaded, 60_000)); // 默认 60s TTL
return loaded;
}
为什么不用 WeakReference 或 SoftReference 做自动回收
初学者常误以为用 WeakHashMap 或包装 SoftReference 就能“自动释放缓存”,实际效果往往不符合预期:
WeakHashMap的 key 是弱引用,一旦外部无强引用,GC 后整个 entry 就消失——但缓存 key 通常就是字符串字面量或池化对象,几乎永不被回收SoftReference的回收时机由 JVM 内存压力决定,不是按访问频次或 TTL,无法预测;且 JDK 8+ 后策略更保守,容易长期驻留导致 OOM- 两者都不提供并发安全保证,需额外同步,反而抵消了引用类型带来的收益
避免踩 LinkedHashMap 的坑:重写 removeEldestEntry 不等于 LRU 缓存
LinkedHashMap 确实可通过重写 removeEldestEntry 控制大小,但它默认是插入顺序,不是访问顺序——除非构造时传入 true:
Map<String, Object> lruCache = new LinkedHashMap<>(16, 0.75f, true) {
@Override
protected boolean removeEldestEntry(Map.Entry<String, Object> eldest) {
return size() > 1000; // 超过 1000 个就淘汰最久未访问的
}
};
但注意:LinkedHashMap 非线程安全,多线程下必须外层加锁(如 synchronized),而锁粒度大时会严重拖慢性能;另外,它不支持 TTL,过期逻辑仍需手动维护。
真正需要过期 + 并发 + LRU 时,直接用 Caffeine
如果项目已引入 Maven,且缓存需求超出“简单”范畴(例如:毫秒级 TTL、权重驱逐、统计命中率、异步刷新),别自己造轮子。Caffeine 是目前 Java 生态最成熟的本地缓存库,API 清晰,性能远超 Guava Cache:
Cache<String, Object> cache = Caffeine.newBuilder()
.maximumSize(10_000)
.expireAfterWrite(10, TimeUnit.MINUTES)
.recordStats()
.build();
Object value = cache.get("key", k -> loadFromDB(k));
它的 get 方法是原子的,内部用分段锁 + CAS 实现高并发;recordStats() 开启后可随时调用 cache.stats() 查看命中率——这些细节自己实现极易出错。
复杂点在于:Caffeine 的驱逐不是实时触发的,而是惰性清理(在读写时顺带处理),所以内存占用略高于理论值;若对内存敏感,需定期调用 cleanUp(),但不要过于频繁,否则影响吞吐。
今天关于《Java简单缓存实现方法解析》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
170 收藏
-
324 收藏
-
439 收藏
-
351 收藏
-
218 收藏
-
134 收藏
-
106 收藏
-
133 收藏
-
454 收藏
-
230 收藏
-
318 收藏
-
407 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习