0

0

Java中的LinkedHashMap的访问顺序特性_实现LRU缓存淘汰算法的基石

P粉602998670

P粉602998670

发布时间:2026-02-14 12:39:10

|

451人浏览过

|

来源于php中文网

原创

accessorder 必须显式设为 true 才启用访问顺序,使 get() 或 put() 已存在 key 时将对应 entry 移至链表尾部;默认 false 按插入顺序维护,lru 失效。

java中的linkedhashmap的访问顺序特性_实现lru缓存淘汰算法的基石

LinkedHashMap 的 accessOrder 参数怎么设才启用访问顺序

不设 accessOrder 或设为 false(默认),LinkedHashMap 按插入顺序维护节点;只有显式传入 true 才开启访问顺序——即每次调用 get()put() 已存在 key 时,对应 Entry 会被移到链表尾部。

  • 构造时必须写全参: new LinkedHashMap(initialCapacity, loadFactor, true)
  • 漏掉第三个参数或传 falseget() 不会改变顺序,LRU 逻辑直接失效
  • 注意:JDK 8+ 中 computeIfPresent()merge() 等方法也会触发访问重排序,但 replace(K,V) 不会(它不改变 value 时甚至不调用 afterNodeAccess()

为什么重写 removeEldestEntry() 是 LRU 的关键开关

LinkedHashMap 本身不自动淘汰,靠子类重写 removeEldestEntry() 在每次插入后决定是否删最老项。这个方法返回 true 才真删——而“最老”在访问顺序下就是链表头节点。

  • 典型写法:
    protected boolean removeEldestEntry(Map.Entry<K,V> eldest) {
        return size() > MAX_SIZE;
    }
  • 别在方法里手动调用 remove(eldest.getKey()):重复删除或破坏迭代器安全
  • 该方法在 put()putAll() 后触发,但 get() 不触发——所以容量控制只响应写入,读多写少场景需额外兜底

get() 返回 null 时会不会影响 LRU 顺序

不会。如果 key 不存在,get() 返回 null,但内部不调用 afterNodeAccess(),链表结构完全不变。

What-the-Diff
What-the-Diff

检查请求差异,自动生成更改描述

下载
  • 这意味着:查不到的 key 不会“刷新热度”,也不会意外挤走其他有效项
  • 但要注意空值歧义:如果业务允许 value 为 null,就不能靠 get() 返回值判断 key 是否存在,得用 containsKey()
  • 若误把 null 当作缺失而反复 put(),会导致同 key 多次插入,链表尾部堆积冗余节点(虽不影响功能,但浪费内存)

并发环境下直接用 LinkedHashMap 实现 LRU 会出什么问题

会丢数据、顺序错乱、甚至 ConcurrentModificationException。因为 LinkedHashMap 非线程安全,其迭代器和链表操作都无同步保障。

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

  • 常见错误:多个线程同时 get() + put(),导致链表指针断裂,后续遍历抛 NullPointerException
  • 不能简单套 Collections.synchronizedMap():它只锁单个方法,size()removeEldestEntry() 之间存在竞态窗口
  • 生产环境建议用 ConcurrentHashMap + 自研双向链表,或直接用 caffeine;非要手写,至少得把整个 get-put-removeEldestEntry 流程包进 synchronized
访问顺序不是默认行为,accessOrder 必须显式设为 true;淘汰逻辑依赖 removeEldestEntry() 的返回值,而不是它的内容;并发场景下,哪怕只读也得小心——因为 get() 在访问顺序模式下会修改内部结构。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

244

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

706

2024.03.01

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

244

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

706

2024.03.01

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

416

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

588

2023.08.10

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

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

673

2023.08.10

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

447

2023.08.14

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

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

23

2026.02.13

热门下载

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

精品课程

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

共23课时 | 3.5万人学习

C# 教程
C# 教程

共94课时 | 9.3万人学习

Java 教程
Java 教程

共578课时 | 64.8万人学习

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

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