Java 实现自定义 LRU 缓存

news2024/11/14 4:05:53

一、引言

在现代软件系统中,缓存是提高性能的重要手段之一。LRU 缓存作为一种常用的缓存策略,能够根据数据的使用频率自动淘汰最近最少使用的数据,从而保持缓存的高效性。在 Java 中,虽然有一些现成的缓存框架可供使用,但了解如何自己实现一个 LRU 缓存可以更好地掌握缓存的原理和优化方法。本文将介绍如何用 Java 实现一个自定义的 LRU 缓存。

二、LRU 缓存概述

(一)LRU 缓存的定义和作用

LRU 缓存是一种按照最近最少使用原则进行数据淘汰的缓存策略。当缓存容量达到上限时,LRU 缓存会自动淘汰最近最少使用的数据,为新的数据腾出空间。LRU 缓存的作用主要有以下几点:

  1. 提高数据访问速度:将经常使用的数据存储在缓存中,可以减少对底层数据源的访问次数,从而提高数据访问速度。
  2. 降低系统负载:通过缓存数据,可以减少对数据库、文件系统等底层数据源的压力,降低系统负载。
  3. 提高系统响应时间:缓存可以快速响应数据请求,减少等待时间,提高系统响应时间。

(二)LRU 缓存的工作原理

LRU 缓存的工作原理基于一个双向链表和一个哈希表。双向链表用于存储缓存中的数据项,按照数据的使用顺序进行排列,最近使用的数据位于链表头部,最近最少使用的数据位于链表尾部。哈希表用于快速查找缓存中的数据项,通过键值对的方式将数据存储在哈希表中。当进行数据访问时,首先在哈希表中查找数据项,如果找到,则将该数据项移动到链表头部,表示最近使用过;如果未找到,则从底层数据源获取数据,并将数据项插入到链表头部和哈希表中。当缓存容量达到上限时,删除链表尾部的数据项,即最近最少使用的数据。

三、Java 实现 LRU 缓存的设计思路

(一)数据结构选择

  1. 双向链表
    • 双向链表是实现 LRU 缓存的关键数据结构之一。它可以方便地实现数据项的插入、删除和移动操作。在 Java 中,可以使用自定义的双向链表类来实现双向链表数据结构。
  2. 哈希表
    • 哈希表用于快速查找缓存中的数据项。在 Java 中,可以使用 HashMap 类来实现哈希表数据结构。

(二)类结构设计

  1. LRUCache 类
    • LRUCache 类是实现 LRU 缓存的核心类。它包含一个双向链表和一个哈希表,用于存储缓存中的数据项。LRUCache 类提供了一些方法,如 put、get、remove 等,用于操作缓存中的数据项。
  2. Node 类
    • Node 类是双向链表中的节点类。它包含一个键值对和指向前一个节点和后一个节点的指针。Node 类用于存储缓存中的数据项,并在双向链表中进行移动操作。

(三)方法设计

  1. put 方法
    • put 方法用于将一个键值对插入到缓存中。如果缓存中已经存在该键,则更新对应的值,并将该节点移动到链表头部;如果缓存中不存在该键,则将新的节点插入到链表头部和哈希表中。如果缓存容量达到上限,则删除链表尾部的节点。
  2. get 方法
    • get 方法用于从缓存中获取一个键对应的值。如果缓存中存在该键,则将该节点移动到链表头部,并返回对应的值;如果缓存中不存在该键,则返回 null。
  3. remove 方法
    • remove 方法用于从缓存中删除一个键值对。如果缓存中存在该键,则删除对应的节点,并从哈希表中移除该键值对;如果缓存中不存在该键,则不进行任何操作。

四、Java 实现 LRU 缓存的具体步骤

(一)定义 Node 类

class Node {
    int key;
    int value;
    Node prev;
    Node next;

    public Node(int key, int value) {
        this.key = key;
        this.value = value;
    }
}

(二)定义 LRUCache 类

import java.util.HashMap;

class LRUCache {
    private int capacity;
    private HashMap<Integer, Node> map;
    private Node head;
    private Node tail;

    public LRUCache(int capacity) {
        this.capacity = capacity;
        map = new HashMap<>();
        head = new Node(0, 0);
        tail = new Node(0, 0);
        head.next = tail;
        tail.prev = head;
    }

    public int get(int key) {
        if (map.containsKey(key)) {
            Node node = map.get(key);
            removeNode(node);
            addToHead(node);
            return node.value;
        } else {
            return -1;
        }
    }

    public void put(int key, int value) {
        if (map.containsKey(key)) {
            Node node = map.get(key);
            node.value = value;
            removeNode(node);
            addToHead(node);
        } else {
            if (map.size() == capacity) {
                Node lastNode = tail.prev;
                removeNode(lastNode);
                map.remove(lastNode.key);
            }
            Node newNode = new Node(key, value);
            addToHead(newNode);
            map.put(key, newNode);
        }
    }

    private void removeNode(Node node) {
        node.prev.next = node.next;
        node.next.prev = node.prev;
    }

    private void addToHead(Node node) {
        node.next = head.next;
        node.prev = head;
        head.next.prev = node;
        head.next = node;
    }
}

(三)测试 LRUCache 类

public class Main {
    public static void main(String[] args) {
        LRUCache cache = new LRUCache(2);
        cache.put(1, 1);
        cache.put(2, 2);
        System.out.println(cache.get(1)); // 输出 1
        cache.put(3, 3);
        System.out.println(cache.get(2)); // 输出 -1
        cache.put(4, 4);
        System.out.println(cache.get(1)); // 输出 -1
        System.out.println(cache.get(3)); // 输出 3
        System.out.println(cache.get(4)); // 输出 4
    }
}

五、LRU 缓存的性能优化

(一)减少哈希表的冲突

  1. 选择合适的哈希函数
    • 选择一个好的哈希函数可以减少哈希表的冲突。在 Java 中,可以使用 Object 的 hashCode 方法作为哈希函数,但需要注意的是,不同的对象可能会产生相同的哈希值,从而导致哈希表的冲突。为了减少冲突,可以对 hashCode 方法的结果进行进一步的处理,如使用取模运算等。
  2. 调整哈希表的容量
    • 调整哈希表的容量也可以减少冲突。如果哈希表的容量过小,容易导致冲突增加;如果哈希表的容量过大,会浪费内存空间。可以根据缓存的容量和预期的负载情况,选择一个合适的哈希表容量。

(二)优化双向链表的操作

  1. 使用高效的链表实现
    • 在 Java 中,可以使用自定义的双向链表类来实现双向链表数据结构。为了提高链表的操作效率,可以使用一些优化技巧,如使用尾指针、避免频繁的内存分配等。
  2. 减少节点的移动次数
    • 在 LRU 缓存中,节点的移动操作比较频繁。为了减少节点的移动次数,可以在节点的属性中增加一个访问计数器,记录节点被访问的次数。当需要淘汰数据时,可以根据访问计数器的值来选择最近最少使用的节点,而不是直接选择链表尾部的节点。

(三)并发访问的处理

  1. 使用线程安全的容器
    • 如果 LRU 缓存需要在多线程环境下使用,可以使用线程安全的容器来代替 HashMap 和自定义的双向链表。在 Java 中,可以使用 ConcurrentHashMap 和 ConcurrentLinkedDeque 等线程安全的容器来实现 LRU 缓存。
  2. 加锁机制
    • 如果不能使用线程安全的容器,可以通过加锁机制来保证 LRU 缓存的线程安全。在 Java 中,可以使用 synchronized 关键字或 ReentrantLock 等锁来实现加锁机制。但需要注意的是,加锁会降低并发性能,因此需要谨慎使用。

六、实际应用案例分析

(一)案例背景

假设有一个电商系统,需要缓存商品信息以提高查询性能。商品信息的查询频率较高,但商品的数量也比较多,因此需要使用 LRU 缓存来管理商品信息的缓存。

(二)缓存设计

  1. 缓存容量的确定
    • 根据系统的负载情况和内存限制,确定 LRU 缓存的容量。如果缓存容量过小,容易导致缓存命中率低;如果缓存容量过大,会浪费内存空间。可以通过性能测试和监控来调整缓存容量。
  2. 缓存数据的存储结构
    • 商品信息可以用一个对象来表示,包含商品的 ID、名称、价格、库存等属性。可以将商品信息对象作为 LRU 缓存中的值,商品的 ID 作为键。在 LRUCache 类中,可以使用一个 HashMap 来存储键值对,使用一个双向链表来维护数据的使用顺序。

(三)缓存的使用

  1. 查询商品信息
    • 当需要查询商品信息时,首先在 LRU 缓存中查找。如果缓存中存在该商品信息,则直接返回;如果缓存中不存在,则从数据库中查询,并将查询结果插入到缓存中。
  2. 更新商品信息
    • 当商品信息发生变化时,需要更新缓存中的数据。可以先从缓存中删除旧的商品信息,然后将新的商品信息插入到缓存中。
  3. 缓存的淘汰
    • 当缓存容量达到上限时,LRU 缓存会自动淘汰最近最少使用的商品信息。可以通过监控缓存的使用情况,及时调整缓存容量,以保证缓存的命中率。

(四)性能优化

  1. 减少数据库查询次数
    • 通过缓存商品信息,可以减少对数据库的查询次数,从而提高系统的性能。可以通过监控缓存的命中率,评估缓存的效果,并根据实际情况进行调整。
  2. 优化缓存的淘汰策略
    • 可以根据商品的访问频率和更新频率,调整 LRU 缓存的淘汰策略。例如,可以对访问频率较高的商品进行特殊处理,避免被过早淘汰。
  3. 并发访问的处理
    • 如果电商系统是一个高并发的系统,需要考虑 LRU 缓存的并发访问问题。可以使用线程安全的容器来实现 LRU 缓存,或者通过加锁机制来保证缓存的线程安全。

七、总结

本文介绍了如何用 Java 实现一个自定义的 LRU 缓存。通过对 LRU 缓存的原理、设计思路、实现步骤以及性能优化的详细介绍,为 Java 技术专家和架构师提供了全面的 LRU 缓存实现指南。在实际应用中,可以根据具体的需求和场景,对 LRU 缓存进行适当的调整和优化,以提高系统的性能和可扩展性。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

qt QVideoWidget详解

1. 概述 QVideoWidget是Qt框架中用于视频播放的控件。它继承自QWidget&#xff0c;并提供了与QMediaPlayer等多媒体播放类集成的功能。QVideoWidget可以嵌入到Qt应用程序的用户界面中&#xff0c;用于显示视频内容。它支持多种视频格式&#xff0c;并提供了基本的视频播放控制…

PG逻辑复制的REPLICA IDENTITY几种设置

前两天同事问了一个PG的错误&#xff0c;创建一张普通表&#xff0c;insert插入正常&#xff0c;但是执行update和delete时&#xff0c;提示这个错误&#xff0c; 代码语言&#xff1a;javascript 复制 SQL 错误 [55000]: ERROR: cannot delete from table "temp_tb&qu…

Flutter 小技巧之 Shader 实现酷炫的粒子动画

在之前的《不一样的思路实现炫酷 3D 翻页折叠动画》我们其实介绍过&#xff1a;如何使用 Shader 去实现一个 3D 的翻页效果&#xff0c;具体就是使用 Flutter 在 3.7 开始提供 Fragment Shader API &#xff0c;因为每个像素都会过 Fragment Shader &#xff0c;所以我们可以通…

<项目代码>YOLOv7 草莓叶片病害识别<目标检测>

YOLOv7是一种单阶段&#xff08;one-stage&#xff09;检测算法&#xff0c;它将目标检测问题转化为一个回归问题&#xff0c;能够在一次前向传播过程中同时完成目标的分类和定位任务。相较于两阶段检测算法&#xff08;如Faster R-CNN&#xff09;&#xff0c;YOLOv7具有更高的…

一文读懂什么是RAG?附MindSpore和MindNLP实现的TinyRAG框架

什么是RAG&#xff1f; 首先我们给出RAG的定义&#xff1a;RAG&#xff08;Retrieval-Augmented Generation&#xff09;技术是一种结合了信息检索&#xff08;Retrieval&#xff09;和生成式模型&#xff08;Generation&#xff09;的人工智能方法。对于用户的Query&#xff…

字节、快手、Vidu“打野”升级,AI视频小步快跑

文&#xff5c;白 鸽 编&#xff5c;王一粟 继9月份版本更新之后&#xff0c;光锥智能从生数科技联合创始人兼CEO唐家渝朋友圈获悉&#xff0c;Vidu大模型将于本周再次进行版本升级&#xff0c;Vidu-1.5版本即将上线。 此版本更新方向仍是重点延伸大模型的泛化能力和主体…

matlab建模入门指导

本文以水池中鸡蛋温度随时间的变化为切入点&#xff0c;对其进行数学建模并进行MATLAB求解&#xff0c;以更为通俗地进行数学建模问题入门指导。 一、问题简述 一个煮熟的鸡蛋有98摄氏度&#xff0c;将它放在18摄氏度的水池中&#xff0c;五分钟后鸡蛋的温度为38摄氏度&#x…

React Query在现代前端开发中的应用

&#x1f493; 博客主页&#xff1a;瑕疵的CSDN主页 &#x1f4dd; Gitee主页&#xff1a;瑕疵的gitee主页 ⏩ 文章专栏&#xff1a;《热点资讯》 React Query在现代前端开发中的应用 React Query在现代前端开发中的应用 React Query在现代前端开发中的应用 引言 React Query …

汇总常用的114款AI视频创作工具,堪称运营神器,收藏备用!

随着AI工具的使用起来起广泛&#xff0c;国内各个互联网大厂都开始在圈内出围。过去我们写文案、做视频、拍视频、剪辑视频、画漫画、处理图片等&#xff0c;都需要手工一点一点地精雕细琢。现在通过AI工具&#xff0c;零基础也能做出很多精致的作品。 前面我在上个月的28号分…

在vue中,完成@wangeditor/editor组件的大数据量加载,解决卡顿

背景 简单说一下需求&#xff0c;一个页面中只存在一个Editor组件&#xff0c;但是需要通过选择不同类型展示不同的content的数据&#xff0c;不过直接通过提供的Editor组件加载的时候&#xff0c;在数据量大&#xff08;测试数据226KB&#xff09;的情况下&#xff0c; 切换类…

通义千问API调用测试 (colab-python,vue)

文章目录 代码&#xff08;来自官网&#xff09;colab中用python测试Qwen2.5在官网上查看并确定过期时间这里看到我的免费额度到25年5月在同一个页面&#xff0c;点击API示例 前端调用直接在前端调用的优缺点以vue为例&#xff08;代码是基于官网node.js的代码转换而来&#xf…

使用 Elasticsearch 构建食谱搜索(一)

作者&#xff1a;来自 Elastic Andre Luiz 了解如何使用 Elasticsearch 构建基于语义搜索的食谱搜索。 简介 许多电子商务网站都希望增强其食谱搜索体验。正确使用语义搜索可以让客户根据更自然的查询&#xff08;例如 “something for Valentines Day - 情人节的礼物” 或 “…

微服务各组件整合

nacos 第一步&#xff0c;引入依赖 <dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId></dependency> 第二步&#xff0c;增加配置 spring:application:name: …

【大数据学习 | HBASE高级】hive操作hbase

一般在查询hbase的数据的时候我们可以直接使用hbase的命令行或者是api进行查询就行了&#xff0c;但是在日常的计算过程中我们一般都不是为了查询&#xff0c;都是在查询的基础上进行二次计算&#xff0c;所以使用hbase的命令是没有办法进行数据计算的&#xff0c;并且对于hbas…

modbus协议 Mthings模拟器使用

进制转换 HEX 16进制 (0、1、2、3、4、5、6、7、8、9、A、B、C、D、E、F表示0-15) dec 10进制 n(16进制) -> 10 abcd.efg(n) d*n^0 c*n^1 b*n^2 a*n^3 e*n^-1 f*n^-2 g*n^-3&#xff08;10&#xff09; 10 -> n(16进制) Modbus基础概念 高位为NUM_H&…

列表(list)

一、前言 本次博客主要讲解 list 容器的基本操作、常用接口做一个系统的整理&#xff0c;结合具体案例熟悉自定义内部排序方法的使用。如有任何错误&#xff0c;欢迎在评论区指出&#xff0c;我会积极改正。 二、什么是list list是C的一个序列容器&#xff0c;插入和删除元素…

Sam Altman:年底将有重磅更新,但不是GPT-5!

大家好&#xff0c;我是木易&#xff0c;一个持续关注AI领域的互联网技术产品经理&#xff0c;国内Top2本科&#xff0c;美国Top10 CS研究生&#xff0c;MBA。我坚信AI是普通人变强的“外挂”&#xff0c;专注于分享AI全维度知识&#xff0c;包括但不限于AI科普&#xff0c;AI工…

zabbix监控端界面时间与服务器时间不对应

1. 修改系统时间 # tzselect Please select a continent, ocean, "coord", or "TZ".1) Africa2) Americas3) Antarctica4) Asia5) Atlantic Ocean6) Australia7) Europe8) Indian Ocean9) Pacific Ocean 10) coord - I want to use geographical coordina…

大数据新视界 -- 大数据大厂之 Impala 性能提升:高级执行计划优化实战案例(下)(18/30)

&#x1f496;&#x1f496;&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎你们来到 青云交的博客&#xff01;能与你们在此邂逅&#xff0c;我满心欢喜&#xff0c;深感无比荣幸。在这个瞬息万变的时代&#xff0c;我们每个人都在苦苦追寻一处能让心灵安然栖息的港湾。而 我的…

P2356 弹珠游戏

铁子们好呀&#xff0c;博主好久没更新了&#xff0c;今天给大家更新一道编程题&#xff01;&#xff01;&#xff01; 题目链接如下&#xff1a;P2356 弹珠游戏 好&#xff0c;接下来&#xff0c;我将从三个方面讲解这道例题。分别是 题目解析算法原理代码实现 文章目录 1.题…