使用LinkedHashMap实现固定大小的LRU缓存

2024-08-27 06:04

本文主要是介绍使用LinkedHashMap实现固定大小的LRU缓存,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

使用LinkedHashMap实现固定大小的LRU缓存

1. 什么是LRU?

LRU是"Least Recently Used"的缩写,意为"最近最少使用"。LRU缓存是一种常用的缓存淘汰算法,它的核心思想是:当缓存满时,优先淘汰最近最少使用的项目。

LRU缓存的工作原理:

  1. 新数据插入到缓存头部
  2. 每当缓存命中(即缓存数据被访问),则将数据移到缓存头部
  3. 当缓存满时,将链表尾部的数据丢弃

LRU算法的理论基础:

LRU算法基于"时间局部性原理"(Principle of Temporal Locality),该原理指出,如果一个信息项正在被访问,那么在近期它很可能还会被再次访问。这一原理在计算机科学中广泛应用,例如在操作系统的页面置换算法中。

LRU的应用场景:

  1. 数据库缓存:减少对数据库的直接访问,提高查询速度
  2. Web应用:缓存经常访问的页面或数据
  3. 硬件设计:CPU缓存的替换策略
  4. 操作系统:页面置换算法

2. LinkedHashMap与LRU缓存

LinkedHashMap的特性:

LinkedHashMap是Java集合框架中的一个类,它继承自HashMap,但在内部维护了一个双向链表,用于保持插入顺序或访问顺序。

关键特性:

  1. 可选的排序模式:插入顺序(默认)或访问顺序
  2. 预测遍历顺序:可以按照特定顺序遍历元素
  3. 性能:大部分操作的时间复杂度为O(1)

LinkedHashMap如何支持LRU:

LinkedHashMap通过以下机制支持LRU缓存的实现:

  1. 访问顺序:通过构造函数的accessOrder参数设置为true,启用访问顺序模式
  2. 自动重排序:每次访问元素时,该元素会被移到链表末尾(最近使用)
  3. removeEldestEntry方法:允许在插入新元素时,决定是否删除最老的元素

继承LinkedHashMap并重写removeEldestEntry方法:

要实现LRU缓存,我们需要:

  1. 创建一个新类,继承LinkedHashMap
  2. 在构造函数中,设置LinkedHashMap的访问顺序为true
  3. 重写removeEldestEntry方法,当map中的元素个数超过指定容量时返回true

3. 代码实现与深入分析

代码实现:

以下是一个简洁的LRU缓存实现,包含了基本功能和性能监控:

LRUCache.java
import java.util.LinkedHashMap;
import java.util.Map;/*** @desc: 使用LinkedHashMap自定义LRU缓存实现* @author: shy* @date: 2024/08/26 10:03*/
public class LRUCache<K, V> extends LinkedHashMap<K, V> {private final int capacity;// 命中数(性能监控)private int hits = 0;// 未命中数(性能监控)private int misses = 0;public LRUCache(int capacity) {super(capacity, 0.75f, true);this.capacity = capacity;}@Overrideprotected boolean removeEldestEntry(Map.Entry<K, V> eldest) {return size() > capacity;}@Overridepublic V get(Object key) {V value = super.get(key);if (value != null) {hits++;} else {misses++;}return value;}public double getHitRate() {int total = hits + misses;return total == 0 ? 0 : (double) hits / total;}
}
MapTest.java
public class MapTest {public static void main(String[] args) {LRUCache<Integer, String> cache = new LRUCache<>(3);cache.put(1, "one");cache.put(2, "two");cache.put(3, "three");System.out.println(cache); // 输出: {1=one, 2=two, 3=three}cache.get(1);System.out.println(cache); // 输出: {2=two, 3=three, 1=one}cache.put(4, "four");System.out.println(cache); // 输出: {3=three, 1=one, 4=four}// 输出缓存命中率System.out.println("Hit rate: " + cache.getHitRate());}
}
执行结果

在这里插入图片描述

代码分析:

  1. 简洁实现:通过继承LinkedHashMap,我们只需要很少的代码就能实现LRU缓存的核心功能。
  2. 容量控制:重写removeEldestEntry方法,确保缓存大小不超过指定容量。
  3. 访问顺序:在构造函数中设置accessOrder为true,确保元素按访问顺序排列。
  4. 性能监控:添加了简单的命中率计算功能,有助于评估缓存效果。
  5. 泛型支持:使用泛型实现,增加了代码的灵活性和复用性。

4. LinkedHashMap实现LRU的优势与劣势

优势:

  1. 实现简单:

    • 利用Java标准库,无需额外依赖
    • 代码量少,易于理解和维护
  2. 性能较好:

    • 大多数操作时间复杂度为O(1)
    • 内部使用哈希表,提供快速的查找性能
  3. 功能完整:

    • 自动维护访问顺序
    • 支持快速的插入和删除操作
  4. 灵活性:

    • 可以轻松扩展,添加自定义功能(如上面的命中率计算)
    • 支持泛型,可用于各种数据类型

劣势:

  1. 内存占用:

    • 比普通HashMap占用更多内存,因为需要维护双向链表
    • 对于大容量缓存,可能会成为性能瓶颈
  2. 并发性能:

    • 默认非线程安全,在多线程环境下需要额外的同步机制
    • 全局同步可能导致高并发场景下的性能问题
  3. 功能局限:

    • 不支持过期时间等高级特性
    • 缺乏分布式缓存支持
  4. 扩展性限制:

    • 继承自LinkedHashMap,可能限制了与其他类的集成
    • 在复杂系统中,可能需要更灵活的接口设计

5. 实际应用中的注意事项

  1. 缓存大小选择:

    • 需要根据实际应用场景和可用内存来确定
    • 考虑缓存命中率和系统性能的平衡
  2. 并发处理:

    • 在多线程环境中,需要注意同步问题
    • 考虑使用 Collections.synchronizedMap() 包装 LRUCache,或使用 ConcurrentHashMap 的变体
  3. 缓存预热:

    • 在系统启动时,可以预先加载常用数据到缓存中
    • 有助于提高系统初期的响应速度
  4. 缓存一致性:

    • 当底层数据发生变化时,需要及时更新或失效缓存
    • 考虑实现缓存更新策略(如写透、延迟写入等)
  5. 监控和调优:

    • 实现缓存命中率、占用空间等指标的监控
    • 根据监控数据定期调整缓存策略

6. 替代方案和进阶技巧

  1. Guava Cache:

    • Google的Guava库提供了更强大的缓存实现
    • 支持过期时间、自动加载、最大大小限制等特性
  2. Caffeine:

    • 高性能的Java缓存库,在许多方面超越了Guava Cache
    • 提供了更灵活的配置选项和更好的并发性能
  3. 多级缓存:

    • 结合内存缓存和分布式缓存(如Redis)
    • 可以平衡访问速度和数据容量
  4. 自定义驱逐策略:

    • 除LRU外,还可以实现LFU(最不经常使用)、FIFO等策略
    • 根据实际应用需求选择或组合不同的策略
  5. hutool-cache:

    • 功能丰富的缓存工具类
    • 支持设置缓存的过期时间和最大容量
    • 支持灵活地控制缓存的生命周期和大小

通过使用LinkedHashMap实现固定大小的LRU缓存的实现,展示了如何使用LinkedHashMap创建一个简单而有效的LRU缓存。这个实现保持了代码的简洁性,同时仍然提供了基本的性能监控功能。在实际应用中,可以根据具体需求进行进一步的扩展和优化。

这篇关于使用LinkedHashMap实现固定大小的LRU缓存的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1110840

相关文章

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

Hadoop数据压缩使用介绍

一、压缩原则 (1)运算密集型的Job,少用压缩 (2)IO密集型的Job,多用压缩 二、压缩算法比较 三、压缩位置选择 四、压缩参数配置 1)为了支持多种压缩/解压缩算法,Hadoop引入了编码/解码器 2)要在Hadoop中启用压缩,可以配置如下参数

Makefile简明使用教程

文章目录 规则makefile文件的基本语法:加在命令前的特殊符号:.PHONY伪目标: Makefilev1 直观写法v2 加上中间过程v3 伪目标v4 变量 make 选项-f-n-C Make 是一种流行的构建工具,常用于将源代码转换成可执行文件或者其他形式的输出文件(如库文件、文档等)。Make 可以自动化地执行编译、链接等一系列操作。 规则 makefile文件

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo