0

0

在Java里如何使用LinkedHashMap实现LRU缓存_Java缓存实现解析

P粉602998670

P粉602998670

发布时间:2026-02-13 11:33:48

|

515人浏览过

|

来源于php中文网

原创

linkedhashmap 能直接支持 lru 是因为它内置访问顺序模式(accessorder = true),使 get() 和 put() 操作自动将节点移至链表尾部,配合重写 removeeldestentry() 可在超容时移除头节点(最久未用项)。

在java里如何使用linkedhashmap实现lru缓存_java缓存实现解析

为什么 LinkedHashMap 能直接支持 LRU?

因为 LinkedHashMap 内置了访问顺序模式(accessOrder = true),它会在每次 get()put() 后把对应节点移到链表尾部;而默认的插入顺序模式(accessOrder = false)只按插入顺序维护——LRU 正好需要「最近访问在尾,最久未用在头」的结构。

关键点在于:必须用带四个参数的构造函数显式启用访问顺序,否则即使重写 removeEldestEntry() 也无效。

示例初始化:

Map<Integer, String> cache = new LinkedHashMap<>(16, 0.75f, true) {
    @Override
    protected boolean removeEldestEntry(Map.Entry<Integer, String> eldest) {
        return size() > 10; // 容量超 10 时移除头节点(最久未用)
    }
};

如何确保 put() 和 get() 都触发 LRU 更新?

默认情况下,get()accessOrder = true 下会更新位置,但 put() 的行为需注意:

立即学习Java免费学习笔记(深入)”;

  • 如果 key 已存在,put() 会覆盖 value 并将该 entry 移至尾部(符合 LRU)
  • 如果 key 不存在,put() 插入新 entry 到尾部(也符合逻辑)
  • 但若用 putIfAbsent()computeIfAbsent(),它们不会触发位置更新 —— 这些方法绕过了 LinkedHashMap 的 access-order 机制

所以实际使用中,应统一走 put() + get(),避免混用非标准更新方法。

腾讯云AI代码助手
腾讯云AI代码助手

基于混元代码大模型的AI辅助编码工具

下载

removeEldestEntry() 的触发时机和常见陷阱

removeEldestEntry() 只在每次 put()(含 putAll())之后被调用一次,**不会在 get() 后触发**。这意味着:

  • 缓存大小只在写入时检查并收缩,读多写少场景下可能短暂超限
  • 返回 true 时,LinkedHashMap 会自动移除头节点(即最久未用项)
  • 不要在该方法里调用 size() 以外的 map 操作(如 get()remove()),会引发 ConcurrentModificationException
  • 若想支持动态容量,可在外部封装一层,把容量作为成员变量传入匿名类,或改用静态内部类实现

线程安全问题怎么处理?

LinkedHashMap 本身不保证线程安全。多线程环境下直接使用会导致:

  • ConcurrentModificationException(遍历时被其他线程修改)
  • 数据错乱(如两个线程同时 put() 导致链表断裂)

常见应对方式:

  • Collections.synchronizedMap(new LinkedHashMap(...)) —— 简单但粒度粗(整个 map 锁),高并发下性能差
  • 自己加 synchronized 块包裹所有操作(get/put/size)—— 更可控,但易漏
  • 改用 ConcurrentHashMap + 手动维护访问顺序(复杂,一般不推荐)

真正需要高性能且线程安全的 LRU 缓存,建议直接用 caffeineguava Cache,它们底层已做精细优化。

LinkedHashMap 实现 LRU 的核心就三件事:开 accessOrder、重写 removeEldestEntry、避开非标准更新方法——其余都是边界细节。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
guava包作用
guava包作用

guava是一个java库,增强了java标准库,提供更有效率和易于使用的集合、实用程序、缓存和并发工具。想了解更多guava的相关内容,可以阅读本专题下面的文章。

265

2024.05.29

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

653

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

325

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

23

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

24

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

90

2026.02.06

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

75

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

36

2025.11.16

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

1

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.5万人学习

C# 教程
C# 教程

共94课时 | 9.3万人学习

Java 教程
Java 教程

共578课时 | 64.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号