手撕LRU 最近最少使用缓存淘汰策略 + LinkedHashMap

LRU 最近最少使用缓存淘汰策略

  • 1 LRU 算法就是一种缓存淘汰策略
  • 2 手撕LRU
  • 3 LinkedHashMap 常见面试题

1 LRU 算法就是一种缓存淘汰策略

计算机的缓存容量有限,如果缓存满了就要删除一些内容,给新内容腾位置。但问题是,删除哪些内容呢?我们肯定希望删掉哪些没什么用的缓存,而把有用的数据继续留在缓存里,方便之后继续使用。那么,什么样的数据,我们判定为「有用的」的数据呢?

LRU 缓存淘汰算法就是一种常用策略。LRU 的全称是 Least Recently Used,也就是说认为最近使用过的数据应该是是「有用的」,很久都没用过的数据应该是无用的,内存满了就优先删那些很久没用过的数据

ps:数据库采用改进的LRU 为了解决select *这种只使用一次,但是会把热数据:真正频率较高的挤到后面的问题。

2 手撕LRU

=========参考题解=========

设计并实现一个满足 LRU (最近最少使用) 缓存 约束的数据结构
实现LRUCache类:

  • LRUCache(int capacity) 以 正整数 作为容量 capacity 初始化 LRU 缓存
  • int get(int key) 如果关键字 key 存在于缓存中,则返回关键字的值,否则返回 -1 。
  • void put(int key, int value)
    • 如果关键字 key 已经存在,则变更其数据值 value
    • 如果不存在,则向缓存中插入该组 key-value
    • 如果插入操作导致关键字数量超过 capacity ,则应该 逐出 最久未使用的关键字

函数 get 和 put 必须以 O(1) 的平均时间复杂度运行。

示例:

输入:
[“LRUCache”, “put”, “put”, “get”, “put”, “get”, “put”, “get”, “get”, “get”]
[[2], [1, 1], [2, 2], [1], [3, 3], [2], [4, 4], [1], [3], [4]]

输出:
[null, null, null, 1, null, -1, null, -1, 3, 4]

解释:
LRUCache lRUCache = new LRUCache(2);
lRUCache.put(1, 1); // 缓存是 {1=1}
lRUCache.put(2, 2); // 缓存是 {1=1, 2=2}
lRUCache.get(1); // 返回 1
lRUCache.put(3, 3); // 该操作会使得关键字 2 作废,缓存是 {1=1, 3=3}
lRUCache.get(2); // 返回 -1 (未找到)
lRUCache.put(4, 4); // 该操作会使得关键字 1 作废,缓存是 {4=4, 3=3}
lRUCache.get(1); // 返回 -1 (未找到)
lRUCache.get(3); // 返回 3
lRUCache.get(4); // 返回 4

⭐️要点👿

  1. 定义一个双向链表:链表中的每个Node 包括:key value两个属性双向链表+dummy节点+形成一个环结构
    在这里插入图片描述
public static class Node{  //定义一个双向链表静态内部类 每个节点有key+value,还有prev指针 next指针(实现双向)
       int key;
       int value;
       Node prev;
       Node next;
       Node(int key, int value){ // 构造器
          this.key = key;
          this.value = value;
      }
  }
  1. 定义一个Hashmap:<key, Node>,为了方便根据 key 查找到对应的Node
    在这里插入图片描述
    capacity: 这是LRUCache类中存储缓存容量的变量,应该在对象创建后不可更改,因此使用 private final 来确保其值在对象创建后不会被修改。
    dummy 节点: 哨兵节点是双向链表中的一个特殊节点,它不存储任何实际的数据,仅用于简化链表操作。因为哨兵节点在整个对象生命周期中不会改变,所以将其声明为 private final
    myhashmap哈希表: 这是LRUCache类中存储键值对的哈希表,也应该在对象创建后不可更改。使用 private final 可以确保在对象创建后,哈希表的引用不会指向其他对象。
    总的来说,使用 private final 可以增强代码的安全性和稳定性,防止意外的修改或赋值操作,确保这些重要的成员变量在对象创建后保持不变。
private final int capacity; // 存储上限
private final Node dummy = new Node(0,0); // 新建一个dummy节点
private final HashMap<Integer, Node> myhashmap = new HashMap<>(); // 声明一个hashmap用于存储key-Node

  • int get(int key) 如果关键字 key 存在于缓存中,则返回关键字的值,否则返回 -1 。
    🔴 先去Hashmap中寻找有无存在key 【调用getNode(key)】
    a. 若有则返回对应的Node,再返回Node.value 。与此同时,由于操作了该节点,因此将该节点移动到链表头部。
    b. 若无则返回-1。

  • void put(int key, int value)
    - 如果关键字 key 已经存在,则变更其数据值 value ;
    - 如果不存在,则向缓存中插入该组 key-value 。
    - 如果插入操作导致关键字数量超过 capacity,则应该 逐出 最久未使用的关键字
    🔴先去Hashmap中寻找有无存在key 【调用getNode(key)】
    a. 若有则返回对应的Node,修改Node.value属性,即可实现更改值的目的。与此同时,由于操作了该节点,因此将该节点移动到链表头部。
    b. 如果没有,则新建一个Node节点包含key-Node,并将这个新建的节点放进链表头部。与此同时,在Hashmap中也添加这个<key,Node>

// 最近最少使用 用于页面置换算法,淘汰最近最少使用的页面
class LRUCache {

    public static class Node{  //定义一个双向链表静态内部类 每个节点有key+value,还有prev指针 next指针(实现双向)
        int key;
        int value;
        Node prev;
        Node next;
        Node(int key, int value){ // 构造器
            this.key = key;
            this.value = value;
        }
    }

    private final int capacity; // 存储上限
    private final Node dummy = new Node(0,0); // 新建一个dummy节点
    private final HashMap<Integer, Node> myhashmap = new HashMap<>(); // 声明一个hashmap用于存储key-Node

    // 【初始化】LRUCache的构造方法
    // 参数capacity用来指定LRU缓存的容量,即缓存可以存储的key-value对的数量上限
    public LRUCache(int capacity) { 
        this.capacity = capacity;
        // 构造环 放在这里的原因:在Java中,类字段初始化必须在构造方法内或者直接初始化字段时进行,不能在类的主体外进行语句级别的初始化操作,所以不能把下面两行放到构造器外面。
        dummy.prev = dummy;
        dummy.next = dummy;
    }

     // 如果有对应的key,则返回key对应的value。如果没有就返回-1
    public int get(int key) {
        Node node = getNode(key); // 去尝试拿到这个node
        if(node != null) return node.value;
        else return -1;
    }

    // 如果key已经存在,则变更其数据值value 
    // 如果key不存在,则向缓存中插入该组 key-value
    // 如果插入操作导致键值对数量超过 capacity ,则应该 逐出 最久未使用的关键字
    public void put(int key, int value) { 
        Node node = getNode(key); // 去尝试拿到这个node
        if(node == null) { // 如果没拿到,即key不存在,则向缓存中插入该组 key-value
            Node newNode = new Node(key,value);
            pushFront(newNode); // 添加到顶部
            myhashmap.put(key,newNode); // 添加到hashmap

            if(myhashmap.size() > capacity){ // 如果插入操作导致键值对数量超过 capacity ,则应该 逐出 最久未使用的关键字
                Node backNode = dummy.prev;
                remove(backNode); // 从链表移走最后一个节点
                myhashmap.remove(backNode.key); // 从hashmap移走最后一个节点对应的key-Node键值对
            }
        }
        else{ // 如果拿到,即key已经存在,则变更其数据值value 
            node.value = value;
        }
        
    }


    // 【辅助方法】
    // 1.在hashmap中根据key拿到Node
    private Node getNode(int key){
        if(!myhashmap.containsKey(key)){ // 如果没有返回null
            return null; 
        }
        Node node = myhashmap.get(key); // 如果有就把这个Node找出来
        remove(node);  // 1.把他从原来的链表中删了
        pushFront(node); // 2.把他放到链表最开始
        return node;       // 3.返回这个节点
    }

    // 2.把节点从链表中删了 ——前后节点跳跃过node连接即可
    private void remove(Node node){
        node.prev.next = node.next;
        node.next.prev = node.prev;
    }

    // 3.把节点放到链表最开始
    private void pushFront(Node node){
        node.next = dummy.next;
        dummy.next.prev = node;
        node.prev = dummy;
        dummy.next = node;
    }
}

/**
 * Your LRUCache object will be instantiated and called as such:
 * LRUCache obj = new LRUCache(capacity);
 * int param_1 = obj.get(key);
 * obj.put(key,value);
 */

3 LinkedHashMap 常见面试题

LinkedHashMap 常见面试题
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

权限提升-Windows权限提升篇溢出漏洞宝塔面板BypassCS插件化MSF模块化

知识点 1、Web到Win系统提权-权限差异原因 2、Web到Win系统提权-溢出漏洞&#xff08;MSF&CS&#xff09; 3、Web到Win系统提权-集成软件&#xff08;哥斯拉模块Bypass&#xff09; 章节点&#xff1a; 1、Web权限提升及转移 2、系统权限提升及转移 3、宿主权限提升及转移…

看完秒懂原来接口测试用例设计这么简单!

什么是接口 接口&#xff1a;服务端程序对外提供的一种统一的访问方式&#xff0c;通常采用 HTTP协议&#xff0c;通过 不同的url&#xff0c;不同的请求类型&#xff08;GET、POST&#xff09;&#xff0c; 不同的参数&#xff0c;来执行不同的业务逻辑。 客户端大多数的业务…

MySQL关联查询如何优化

好久不见&#xff0c;关于这篇文章&#xff0c;我也是想了很久&#xff0c;还是决定写一篇文章&#xff0c;有很多同学问过 mysql 相关的问题&#xff0c;其实关联查询如何优化&#xff0c;首先我们要知道关联查询的原理是什么&#xff1f; 左连接 left join SELECT 字段列表…

软件测试面试,你准备好了吗?

最近有机会做一些面试工作&#xff0c;主要负责面试软件测试人员招聘的技术面试。 之前一直是应聘者的角色&#xff0c;经历了不少次的面试之后&#xff0c;多少也积累一点面试的经验&#xff0c;现在发生了角色转变。初次的面试就碰到个工作年限比我长的&#xff0c;也没有时…

北斗卫星在公路养护中的应用

北斗卫星在公路养护中的应用 北斗卫星是我国自主研发的一款卫星导航系统&#xff0c;它为公路养护工作提供了新的解决方案。通过使用北斗卫星技术&#xff0c;公路养护部门可以实时获取道路状况&#xff0c;提高工作效率&#xff0c;为交通安全保驾护航。 首先&#xff0c;北斗…

Java使用工厂方法实现聚合调用不同第三方接口进行实名验证

在Java中使用工厂方法实现聚合实名验证指的是创建一种实名验证服务&#xff0c;可以连接到不同的实名验证处理器&#xff0c;比如阿里、腾讯等。我们可以定义一个实名验证接口&#xff0c;然后实现不同的实名验证方式&#xff0c;最后使用一个工厂来创建相应的实名验证实例。以…

MySQL实现事务隔离的秘诀之锁

在MySQL中&#xff0c;有多种锁类型&#xff0c;我们先了解三种概念的锁&#xff0c;以便对接下来的内容有更好理解。 表级锁&#xff08;Table Lock&#xff09;&#xff1a;对整个表加锁&#xff0c;其他事务无法修改或读取该表的数据&#xff0c;但可以对其他表进行操作。页…

SpringCloud入门(1) Eureka Ribbon Nacos

这里写目录标题 认识微服务SpringCloud 服务拆分和远程调用服务拆分案例实现远程调用 RestTemplate Eureka注册中心Eureka的结构和作用搭建eureka-server服务注册服务发现 Ribbon负载均衡 LoadBalancedLoadBalancerIntercepor源码解析负载均衡策略饥饿加载 Nacos注册中心安装与…

Java通过SSH连接数据库

一、实现思路 1 实现思路&#xff1a;本地–>跳板机–>目标数据库 2 IP走向&#xff1a;127.0.0.1:5432 --> 192.168.1.111 -->10.11.12.13:5432 二、引入maven <dependency><groupId>com.jcraft</groupId><artifactId>jsch</artifa…

大规模电商平台数据采集难点分析♫

▁▃▅▇主要包括以下几方面&#xff1a; API工具 ◆◆数据量巨大 任何系统&#xff0c;在不同的数据量面前&#xff0c;需要的技术难度都是完全不同的。 如果单纯是将数据采到&#xff0c;可能还比较好完成&#xff0c;但采集之后还需要处理&#xff0c;因为必须考虑数据的规…

多模态数据融合简介#翻译

翻译自—— 感谢外国友人分享&#xff0c;鄙人在此翻译分享给大家INTRODUCTION TO DATA FUSION. multi-modality | by Haylat T | Haileleol Tibebu | Medium 多模态梳理_多模态图像和多模态方法的区别-CSDN博客 #这个网u也写得不错&#xff01; 多模态 神经网络是最著名的机…

申元智能邀您参观2024长三角快递物流供应链与技术装备展览会

2024年7月8-10日 | 杭州国际博览中心 展会介绍 2024长三角快递物流供应链与技术装备展览会&#xff08;杭州&#xff09;&#xff0c;于2024年7月8-10日在杭州国际博览中心召开&#xff0c;本届展会致力于全面展示快递物流上下游领域的创新解决方案&#xff0c;涵盖快递物流供…

接雨水-热题 100?-Lua 中文代码解题第4题

接雨水-热题 100&#xff1f;-Lua 中文代码解题第4题 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 示例 1&#xff1a; 输入&#xff1a;height [0,1,0,2,1,0,1,3,2,1,2,1] 输出&#xff1a;6 解释…

中型企业网络路由器配置(ensp)实验

vlan、vlan间路由、ospf协议等来实现三层交换机和单臂路由之间的通信 拓扑图&#xff1a; 1. 配置三层交换机vlan和vlan间路由 SW1 #进入视图 sys sysn sw1 undo info-center enable#配置vlan vlan batch 10 20 30 40 50 60#配置access口 int g0/0/1 port link-type access …

第十二届蓝桥杯省赛CC++ 研究生组

十二届省赛题 第十二届蓝桥杯省赛C&C 研究生组-卡片 第十二届蓝桥杯省赛C&C 研究生组-直线 第十二届蓝桥杯省赛C&C 研究生组-货物摆放 第十二届蓝桥杯省赛C&C 研究生组-路径 第十二届蓝桥杯省赛C&C 研究生组-时间显示 第十二届蓝桥杯省赛C&C 研究生组…

数字资产管理系统、企业数字资产管理软件

数字资产管理系统&#xff08;DAMS&#xff09;是一系列软件&#xff0c;它提供了一个开放平台&#xff0c;支持对多媒体数据的采集、创建、管理、存储、归档、检索、传输和显示。这些多媒体数据包括图像、视频、声音、文本和电影剪辑等。这些基础软件不仅是内容创作&#xff0…

普洛斯怀来数据中心获Uptime MO认证,以高品质服务持续提升客户体验

近日&#xff0c;普洛斯怀来数据中心顺利通过Uptime M&O&#xff08;运维与管理&#xff09;认证&#xff0c;获得Uptime Institute颁发的认证证书。普洛斯数据中心致力于为客户提供高品质、高可靠的运维服务&#xff0c;此项认证&#xff0c;标志着普洛斯数据中心运营及管…

基于springboot的班级综合测评管理系统的设计与实现

目录 背景 技术简介 系统简介 界面预览 背景 随着电子技术的广泛渗透和迅猛发展&#xff0c;网络化的管理平台得到了大规模的应用。众多的公共机构和商业组织都在积极推进管理流程的电子化转型&#xff0c;班级的综合评价管理系统亦是如此&#xff0c;从传统的手工操作转变…

移动硬盘故障解析:解决无法访问且位置不可用问题

在我们日常的工作和生活中&#xff0c;移动硬盘已成为存储和传输数据的重要工具。然而&#xff0c;有时我们会遇到移动硬盘无法访问且位置不可用的情况&#xff0c;这无疑给数据的存储和访问带来了极大的困扰。本文将深入探讨这一问题&#xff0c;分析其原因&#xff0c;并给出…

C#事件实例详解

一、什么是事件&#xff1f; 在C#中,事件(event)是一种特殊的类成员,它允许类或对象通知其他类或对象发生了某些事情。 从语法上看,事件的声明类似于字段,但它们在功能和行为上有一些重要的区别。 从技术角度来说,事件实际上是一个封装了事件订阅和取消订阅功能的委托字段。…