使用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缓存。这个实现保持了代码的简洁性,同时仍然提供了基本的性能监控功能。在实际应用中,可以根据具体需求进行进一步的扩展和优化。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/412740.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

18959 二叉树的之字形遍历

### 思路 1. **输入读取**&#xff1a; - 读取输入字符串&#xff0c;表示完全二叉树的顺序存储结构。 2. **构建二叉树**&#xff1a; - 使用队列构建二叉树&#xff0c;按层次顺序插入节点。 3. **之字形层序遍历**&#xff1a; - 使用双端队列进行层序遍历&…

【开端】基于nginx部署的具有网关的web日志分析

一、绪论 基于nginx部署的具有网关的web日志分析&#xff0c;我们可以分析的日志有nginx的access.log &#xff0c;网关的日志和应用的日志 二、日志分析 1、nginx日志 参数 说明 示例 $remote_addr 客户端地址 172.17.0.1 $remote_user 客户端用户名称 -- $time_lo…

简化WPF开发:CommunityToolkit.Mvvm在MVVM架构中的实践与优势

文章目录 前言一、CommunityToolkit.Mvvm1.特点2.优点3.缺点 二、WPF项目应用1.引入到 WPF 项目2.使用示例 总结 前言 CommunityToolkit.Mvvm 是 Microsoft 提供的一个社区工具包&#xff0c;专为 MVVM&#xff08;Model-View-ViewModel&#xff09;模式设计&#xff0c;旨在帮…

RabbitMQ练习(Topics)

1、RabbitMQ教程 《RabbitMQ Tutorials》https://www.rabbitmq.com/tutorials 2、环境准备 参考&#xff1a;《RabbitMQ练习&#xff08;Hello World&#xff09;》和《RabbitMQ练习&#xff08;Work Queues&#xff09;》。 确保RabbitMQ、Sender、Receiver、Receiver2容器…

“重启就能解决一切问题”,iPhone重启方法大揭秘

随着iPhone不断更新换代&#xff0c;其设计与操作方式也在不断进化。从最初的实体Home键到如今的全面屏设计&#xff0c;iPhone的操作逻辑也随之发生了改变。 对于那些习惯了传统安卓手机操作的用户来说&#xff0c;iPhone的重启方式可能会显得有些不同寻常。下面我们就来一起…

SQL血缘解析

Druid 作为使用率特别高的的数据库连接池工具,在具备完善的连接池管理功能外,同时Druid 的 SQL解析功能可以用来防止 SQL注入等安全风险。通过对 SQL 语句进行解析和检查,Druid 可以识别并阻止潜在的恶意 SQL 语句执行,黑名单(阻止特定的 SQL 语句执行)、白名单(仅允许特…

★ 算法OJ题 ★ 力扣11 - 盛水最多的容器

Ciallo&#xff5e;(∠・ω< )⌒☆ ~ 今天&#xff0c;我将和大家一起做一道双指针算法题--盛水最多的容器~ 目录 一 题目 二 算法解析 三 编写算法 一 题目 11. 盛最多水的容器 - 力扣&#xff08;LeetCode&#xff09; 二 算法解析 解法1&#xff1a;暴力枚举 …

文本数据分析-(TF-IDF)(1)

文章目录 一、TF-IDF简介1.意义2.TF与IDF1).TF&#xff08;Term Frequency&#xff09;2).IDF&#xff08;Inverse Document Frequency&#xff09;3).TF-IDF 二、应用三、代码实现1.文件读取2.数据预处理3.排序和输出4.全部代码 一、TF-IDF简介 1.意义 TF-IDF&#xff08;Te…

28 TreeView组件

Tkinter ttk.Treeview 组件使用指南 ttk.Treeview 是 Tkinter 的一个高级控件&#xff0c;用于显示和管理层次化数据。它类似于电子表格或列表视图&#xff0c;但提供了更丰富的功能&#xff0c;如可展开的节点、多列显示等。ttk 模块是 Tkinter 的一个扩展&#xff0c;提供了…

Golang | Leetcode Golang题解之第382题链表随机节点

题目&#xff1a; 题解&#xff1a; type Solution struct {head *ListNode }func Constructor(head *ListNode) Solution {return Solution{head} }func (s *Solution) GetRandom() (ans int) {for node, i : s.head, 1; node ! nil; node node.Next {if rand.Intn(i) 0 { …

《机器学习》数据分析之关键词提取、TF-IDF、项目实现 <下>

目录 一、内容回顾 1、核心算法 2、算法公式 3、拆分文本 二、再次操作 1、取出每一卷的地址和内容 得到下列结果&#xff1a;&#xff08;此为DF类型&#xff09; 2、对每一篇文章进行分词 3、计算TF-IDF值 得到以下数据&#xff1a; 三、总结 1、关键词提取 1&a…

数据挖掘之分类算法

分类算法是数据挖掘中常用的一类算法&#xff0c;其主要任务是根据已知的训练数据&#xff08;即带有标签的数据&#xff09;构建模型&#xff0c;然后利用该模型对新的数据进行分类。分类算法广泛应用于金融、医疗、市场营销等领域&#xff0c;用于预测、决策支持等任务。以下…

STM32G474采用“多个单通道ADC转换”读取3个ADC引脚的电压

STM32G474采用“多个单通道ADC转换”读取3个ADC引脚的电压&#xff1a;PC0、PA1和PA2。本测试将ADC1_IN6映射到PC0引脚&#xff0c;ADC12_IN2映射到PA1引脚&#xff0c;ADC1_IN3映射到PA2引脚。 1、ADC输入 ADC输入电压范围&#xff1a;Vref– ≤ VIN ≤ Vref ADC支持“单端输入…

Java 集合Collection(List、Set)Map

集合的理解和优点 1)可以动态保存任意多个对象&#xff0c;使用比较方便!2)提供了一系列方便的操作对象的方法: add、remove、 set、 get等3)使用集合添加,删除新元素的示意代码- Java集合的分类 Java的集合类很多&#xff0c;主要分为两大类&#xff0c;如图&#xff1a; 1…

iPhone备忘录不小心删除了怎么办?

在日常使用iPhone的过程中&#xff0c;备忘录作为我们记录重要信息、灵感闪现和日常琐事的小帮手&#xff0c;其重要性不言而喻。然而&#xff0c;有时候因为操作失误或是不小心点击&#xff0c;我们可能会将珍贵的备忘录内容删除&#xff0c;这无疑会让人感到焦虑与不安。但请…

深入垃圾回收:理解GC的核心算法与实现

垃圾回收&#xff08;Garbage Collection&#xff0c;GC&#xff09;是现代编程语言中一项关键技术。它不仅解决了内存管理中的诸多问题&#xff0c;还为开发者提供了一个更高效、更安全的编程环境。本文将深入探讨GC的起源、主要算法以及这些算法在不同编程语言中的具体实现。…

考试:计算机网络(01)

网络功能和分类 计算机网络是计算机技术与通信技术相结合的产物&#xff0c;它实现了远程通信、远程信息处理和资源共享。 计算机网络的功能&#xff1a;数据通信、资源共享、管理集中化、实现分布式处理、负载均衡。 网络性能指标&#xff1a;速率、带宽(频带宽度或传送线路…

嵌入式数据库

概述 1.作用&#xff1a;存储大量数据&#xff0c;专业存储数据 存储在内存&#xff08;数组&#xff0c;变量&#xff0c;链表&#xff09;上的特点&#xff1a;程序运行结束&#xff0c;或者掉电&#xff0c;数据会丢失。 存储在硬盘&#xff08;文件&#xff09;上的特点…

vue3+ts+vite项目代码检查报错(vue-tsc)

报错原因&#xff1a;vue-tsc与typescrip版本不兼容 排查流程&#xff1a; 1、开始以为vue-tsc或者typescript版本太低&#xff0c;通过npm update更新&#xff0c;更新后还是报错 2、项目中package.json文件中typescript、vue-tsc版本并无兼容问题 3、控制台执行npm list发…

【HarmonyOS】模仿个人中心头像图片,调用系统相机拍照,从系统相册选择图片和圆形裁剪显示 (一)

【HarmonyOS】头像图片&#xff0c;调用系统相机拍照&#xff0c;从系统相册选择图片和圆形裁剪显示 &#xff08;一&#xff09; Demo效果展示&#xff1a; 方案思路&#xff1a; 使用photoAccessHelper实现系统相册选择图片的功能。此API可在无需用户授权的情况下&#xff…