腾讯一面-LRU缓存

news/2024/12/21 20:35:49/

为了设计一个满足LRU(最近最少使用)缓存约束的数据结构,我们可以使用哈希表(HashMap)来存储键值对,以便在O(1)时间复杂度内访问任意键。同时,我们还需要一个双向链表(Doubly Linked List)来保持键的使用顺序,以便在O(1)时间复杂度内执行插入和删除操作。

我们使用了一个ListNode类来表示双向链表中的节点,每个节点包含键、值、指向前一个节点的指针和指向后一个节点的指针。LRUCache类包含了容量、哈希表、双向链表的头节点和尾节点。

get方法首先检查键是否存在于哈希表中。如果存在,则将对应的节点移到双向链表的头部,并返回节点的值。如果不存在,则返回-1。

put方法首先检查键是否存在于哈希表中。如果不存在,则创建一个新的节点,将其添加到哈希表和双向链表的头部,并检查是否超出了容量。如果超出了容量,则删除双向链表尾部的节点,并从哈希表中移除对应的键值对。如果键已经存在,则更新节点的值,并将其移到双向链表的头部。

addToHeadremoveNodemoveToHeadremoveTail是辅助方法,用于在双向链表中添加节点、删除节点、将节点移到头部和删除尾部节点。

代码如下:

import java.util.HashMap;
import java.util.Map;class ListNode {int key;int value;ListNode prev;ListNode next;ListNode(int k, int v) {key = k;value = v;}
}class LRUCache {private int capacity;private Map<Integer, ListNode> map;private ListNode head;private ListNode tail;public LRUCache(int capacity) {this.capacity = capacity;map = new HashMap<>();head = new ListNode(0, 0);tail = new ListNode(0, 0);head.next = tail;tail.prev = head;}public int get(int key) {ListNode node = map.get(key);if (node == null) {return -1;}moveToHead(node);return node.value;}public void put(int key, int value) {ListNode node = map.get(key);if (node == null) {// 如果 key 不存在,创建一个新的节点ListNode newNode = new ListNode(key, value);// 添加进哈希表map.put(key, newNode);// 添加进双向链表头部addToHead(newNode);// 如果超出容量,删除双向链表尾部节点if (map.size() > capacity) {ListNode tail = removeTail();map.remove(tail.key);}} else {// 如果 key 存在,先通过哈希表定位,再修改 value,并移到头部node.value = value;moveToHead(node);}}private void addToHead(ListNode node) {node.prev = head;node.next = head.next;head.next.prev = node;head.next = node;}private void removeNode(ListNode node) {node.prev.next = node.next;node.next.prev = node.prev;}private void moveToHead(ListNode node) {removeNode(node);addToHead(node);}private ListNode removeTail() {ListNode res = tail.prev;removeNode(res);return res;}
}

addToHead(ListNode node)

这个方法用于将一个新的节点添加到双向链表的头部。

  1. 设置新节点的prev指针:新节点的prev指针指向当前的头节点(head)。

  2. 设置新节点的next指针:新节点的next指针指向头节点的下一个节点(head.next)。

  3. 更新头节点下一个节点的prev指针:因为新节点被插入到了头节点和头节点的下一个节点之间,所以需要更新头节点的下一个节点的prev指针,使其指向新节点。

  4. 更新头节点的next指针:最后,更新头节点的next指针,使其指向新节点,这样新节点就成为了双向链表的新头节点。

removeNode(ListNode node)

这个方法用于从双向链表中移除一个节点。

  1. 更新被移除节点前一个节点的next指针:将被移除节点的prev指针所指向的节点的next指针更新为被移除节点的next指针,这样前一个节点就“跳过”了被移除的节点。

  2. 更新被移除节点后一个节点的prev指针:将被移除节点的next指针所指向的节点的prev指针更新为被移除节点的prev指针,这样后一个节点也“跳过”了被移除的节点。

moveToHead(ListNode node)

这个方法用于将一个已存在的节点从双向链表的当前位置移动到头部。

  1. 调用removeNode方法:首先,使用removeNode方法将被移动的节点从双向链表中移除。

  2. 调用addToHead方法:然后,使用addToHead方法将被移除的节点(现在是游离的)添加到双向链表的头部。

removeTail()

这个方法用于移除双向链表的尾部节点,并返回该节点。

  1. 获取尾部节点的前一个节点:由于尾节点(tail)的prev指针指向了双向链表的最后一个实际存储数据的节点,所以tail.prev就是我们要移除的尾部节点。

  2. 调用removeNode方法:使用removeNode方法移除尾部节点。

  3. 返回被移除的节点:返回被移除的尾部节点。

注意:在removeTail方法中,实际上并没有直接更新tail指针,因为按照LRU缓存的逻辑,尾部节点在移除后通常不需要再被引用。然而,如果出于某种原因需要保持tail指针的有效性(比如在某些实现中,你可能想要保持一个有效的尾部引用以便快速添加新节点到尾部),你可能需要在移除尾部节点后更新tail指针,使其指向新的尾部节点(即原尾部节点的前一个节点)。但在你提供的代码中,这个步骤是省略的,因为tail节点始终是一个哑节点(dummy node),不存储实际数据。


http://www.ppmy.cn/news/1534654.html

相关文章

【算法系列-链表】交换链表节点(反转 + 交换)

【算法系列-链表】交换链表节点(反转 交换) 文章目录 【算法系列-链表】交换链表节点(反转 交换)1. 反转链表1.1 思路分析&#x1f3af;1.2 解题过程&#x1f3ac;1.3 代码示例&#x1f330; 2. 两两交换链表中的节点2.1 思路分析&#x1f3af;2.2 解题过程&#x1f3ac;2.3 …

04DSP学习-利用syscfg配置EPWM

打开syscfg文件&#xff0c;左侧control栏中找到EPWM&#xff0c;点击&#xff0c;发现TI提供了一些帮助文档&#xff0c;帮助了解如何使用syscfg以及如何了解EPWM。我们结合配置过程去理解如何使用。 设计目标 使用EPWM1&#xff1b;增减计数&#xff1b;PWM频率为10kHz&…

多线程-初阶(1)

本节⽬标 • 认识多线程 • 掌握多线程程序的编写 • 掌握多线程的状态 • 掌握什么是线程不安全及解决思路 • 掌握 synchronized、volatile 关键字 1. 认识线程&#xff08;Thread&#xff09; 1.1 概念 1) 线程是什么 ⼀个线程就是⼀个 "执⾏流". 每个线…

深入浅出:现代JavaScript开发者必知必会的Web性能优化技巧

亲爱的读者们&#xff0c;欢迎来到本期博客。今天&#xff0c;我们将深入探讨JavaScript开发者在日常工作中如何提升Web性能。在快节奏的Web开发世界中&#xff0c;性能优化至关重要。本文将分享一些实用技巧&#xff0c;帮助你构建快速、高效的Web应用。 1. 使用CDN加速资源加…

量子计算:颠覆未来计算的革命性技术

量子计算&#xff1a;颠覆未来计算的革命性技术 量子计算作为下一代颠覆性技术&#xff0c;正在引领计算领域的重大变革。与传统计算机基于比特的二进制运算不同&#xff0c;量子计算通过量子比特&#xff08;qubits&#xff09;在叠加态和纠缠态下实现并行计算&#xff0c;能…

C# 入坑JAVA 潜规则 注解 列表 listMch,该列表存储了一个映射(Map)的集合 等 入门系列3

java注解 好像和C# 特性 差不多 Data Builder NoArgsConstructor AllArgsConstructor 在Java中&#xff0c;Data、Builder、NoArgsConstructor和AllArgsConstructor是Lombok库提供的注解&#xff0c;它们用于简化Java对象的创建和处理。Lombok是一个流行的Java库&#xff0c;…

滚雪球学MySQL[2.2讲]:基本数据操作详解:插入、查询、更新与删除

全文目录&#xff1a; 前言2.2 基本数据操作1. 插入数据&#xff08;INSERT&#xff09;基本语法示例1&#xff1a;向所有列插入数据示例2&#xff1a;插入部分列的数据 2. 查询数据&#xff08;SELECT&#xff09;基本语法示例1&#xff1a;查询所有数据示例2&#xff1a;查询…

力扣10.1

983. 最低票价 在一个火车旅行很受欢迎的国度&#xff0c;你提前一年计划了一些火车旅行。在接下来的一年里&#xff0c;你要旅行的日子将以一个名为 days 的数组给出。每一项是一个从 1 到 365 的整数。 火车票有 三种不同的销售方式 &#xff1a; 一张 为期一天 的通行证售…