【力扣】146.LRU缓存

ops/2025/2/13 6:55:10/

AC截图

题目

思路

可以构造一个双向链表,使用dummy作为哨向结点。

①定义结点

class Node{
public:int key;int value;Node* prev;Node* next;Node(int k=0,int v=0):key(k),value(v){}
};

LRUCache类

②定义参数列表

    int capacity;Node* dummy;unordered_map<int,Node*> key_to_node;

③删除节点

    void remove(Node *x){x->prev->next = x->next;x->next->prev = x->prev;}

④将结点置顶

对于插入节点,无论键值是否存在,都需要将该结点置顶,因为如果容量过载,LRU会删除最底部的结点。而该结点现在被访问,应该更新为最新的结点

    void push_front(Node *x){x->prev = dummy;x->next = dummy->next;x->prev->next = x;x->next->prev = x;}

⑤获取结点

如果键不存在,则返回空

如果键存在,则将该结点置顶(先删除,再放入顶部),然后返回结点

    Node* get_node(int key){auto it = key_to_node.find(key);if(it==key_to_node.end()){return NULL;}Node* node = it->second;remove(node);push_front(node);return node;}

⑥LRUCache初始化

    LRUCache(int capacity):capacity(capacity),dummy(new Node()) {dummy->prev = dummy;dummy->next = dummy;}

⑦获取结点

    int get(int key) {Node* node = get_node(key);return node?node->value:-1;}

⑧插入节点

如果键存在,则修改值

如果键不存在:

        以传入的键值构造新结点,更新unordered_map

        将新结点置顶

        判断容量是否过载:

                如果过载,删除最底部结点,更新unordered_map

    void put(int key, int value) {Node* node = get_node(key);if(node){node->value = value;return ;}key_to_node[key] = node = new Node(key,value);push_front(node);if(key_to_node.size()>capacity){Node* back_node = dummy->prev;key_to_node.erase(back_node->key);remove(back_node);delete back_node;}}

完整代码

class Node{
public:int key;int value;Node* prev;Node* next;Node(int k=0,int v=0):key(k),value(v){}
};class LRUCache {
private:int capacity;Node* dummy;unordered_map<int,Node*> key_to_node;void remove(Node *x){x->prev->next = x->next;x->next->prev = x->prev;}void push_front(Node *x){x->prev = dummy;x->next = dummy->next;x->prev->next = x;x->next->prev = x;}Node* get_node(int key){auto it = key_to_node.find(key);if(it==key_to_node.end()){return NULL;}Node* node = it->second;remove(node);push_front(node);return node;}public:LRUCache(int capacity):capacity(capacity),dummy(new Node()) {dummy->prev = dummy;dummy->next = dummy;}int get(int key) {Node* node = get_node(key);return node?node->value:-1;}void put(int key, int value) {Node* node = get_node(key);if(node){node->value = value;return ;}key_to_node[key] = node = new Node(key,value);push_front(node);if(key_to_node.size()>capacity){Node* back_node = dummy->prev;key_to_node.erase(back_node->key);remove(back_node);delete back_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);*/


http://www.ppmy.cn/ops/157985.html

相关文章

npm与包

在 Node.js 的生态系统中&#xff0c;npm&#xff08;Node Package Manager&#xff09;扮演着至关重要的角色。它不仅是管理项目依赖的强大工具&#xff0c;还提供了丰富的第三方库和工具&#xff0c;极大地提高了开发效率。本文将详细介绍 npm 的基本概念、常用命令以及如何创…

9.4双向BFS

一、双向BFS核心思想 双向广度优先搜索&#xff08;Bidirectional BFS&#xff09;是一种优化策略&#xff0c;通过从起点和终点同时进行BFS&#xff0c;在中间相遇时终止搜索。适用于&#xff1a; 明确起点和终点的场景搜索空间较大的最短路径问题分支因子&#xff08;Branc…

开源机器人+具身智能 解决方案+AI

开源机器人、具身智能(Embodied Intelligence)以及AI技术的结合,可以为机器人领域带来全新的解决方案。以下是这一结合的可能方向和具体方案: 1. 开源机器人平台 开源机器人平台为开发者提供了灵活的基础架构,可以在此基础上结合具身智能和AI技术。以下是一些常用的开源机…

网络安全之探险

因为工作相关性&#xff0c;看着第三方公司出具的网络安全和shentou测试报告就想更深入研究一下&#xff0c;于是乎开始探索网络安全方面的知识&#xff0c;度娘、知乎开始一步步开始&#xff0c;总结昨天学到皮毛知识。 1.考证大全&#xff0c;开始是奔着这个目的去的 2.有用…

C++ Primer 条件语句

欢迎阅读我的 【CPrimer】专栏 专栏简介&#xff1a;本专栏主要面向C初学者&#xff0c;解释C的一些基本概念和基础语言特性&#xff0c;涉及C标准库的用法&#xff0c;面向对象特性&#xff0c;泛型特性高级用法。通过使用标准库中定义的抽象设施&#xff0c;使你更加适应高级…

【第三节】CMake 的构建流程

前言 CMake 是一个跨平台的构建工具&#xff0c;广泛用于管理 C 项目的构建过程。它的核心优势在于能够生成适合不同平台的构建文件&#xff08;如 Makefile、Ninja 文件、Visual Studio 工程等&#xff09;&#xff0c;从而简化项目的编译和构建流程。本文将详细介绍 CMake 的…

npm和pnpm的区别

1. 依赖存储机制 npm 默认使用 扁平化结构&#xff08;node_modules&#xff09;&#xff0c;所有依赖直接安装在项目的 node_modules 目录下。 依赖可能存在重复安装&#xff08;尤其是不同版本&#xff09;&#xff0c;导致 磁盘空间浪费。 依赖提升&#xff08;hoisting…

【Linux】tar压缩工具常用参数详解

tar 命令是 Unix/Linux 系统中用于文件打包和压缩的核心工具。它的名字来源于“tape archive”&#xff0c;最初设计用于磁带备份&#xff0c;但现在广泛用于文件归档。tar命令可以将多个文件或目录打包成一个单独的文件&#xff0c;通常称为tar包。之后&#xff0c;还可以使用…