Hot 100 刷题计划】 LeetCode 146. LRU 缓存 | C++ 哈希表+双向链表

LeetCode 146. LRU 缓存

📌 题目描述

题目级别:中等 (实际面试中常作为 Hard 级别压轴)

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

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


💡 破题思路:为什么需要哈希表 + 双向链表?

要想在 O(1) 的时间复杂度内完成查找和更新,我们必须使用两种数据结构的组合:

  1. 哈希表 (unordered_map) :负责满足 get 操作的 O(1) 查找。它能瞬间通过 key 定位到数据在内存中的具体位置。
  2. 双向链表 (Doubly Linked List) :负责维持数据的"时间先后顺序"。
    • 为什么不用单链表?因为当我们通过哈希表命中某个节点,想要把它移到头部时,单链表无法在 O(1) 时间内找到它的前驱节点来断开连接,而双向链表有 pre 指针,可以瞬间"原地拔起"。
    • 为什么不用数组?数组在头部插入或删除元素需要移动大量数据,复杂度是 O(N)。

运作机制:

  • 越靠近链表头部的节点,越是"最近使用"的。
  • 越靠近链表尾部的节点,越是"最久未使用"的。
  • 每次 get 命中一个节点,就把它从原位置拔出来,重新插到头部 (hinert)。
  • 每次 put 新数据,直接插到头部;如果容量超标,就把尾部节点 (tail->pre) 无情淘汰!

极客细节:虚拟头尾节点

人为在链表两端加上 headtail 两个 Dummy 节点,这样任何真实的节点都有前驱和后继,写 removehinert 时再也不用去判断 if (node->pre != nullptr) 这种恶心的边界条件了。


💻 C++ 代码实现

cpp 复制代码
class Node{
public:
    Node *next;
    Node *pre;
    int key, value;
    Node (int k, int v) {
        key = k;
        value = v;
        next = NULL;
        pre = NULL;
    }
};

class LRUCache {
public:
    LRUCache(int capacity) {
        cap = capacity;
        // 初始化虚拟头尾节点,互相指引,彻底消除空指针异常边界
        head = new Node(0, 0);
        tail = new Node(0, 0);
        head -> next = tail;
        tail -> pre = head;
    }
    
    int get(int key) {
        if (mp.count(key))
        {
            // 只要被访问,就说明它是"最近使用"的,立刻把它移到链表头部
            remove(mp[key]);
            hinert(mp[key]);
            return mp[key] -> value;
        }

        return -1;
    }
    
    void put(int key, int value) {
        // 如果 key 已经存在,作者采取了极其果断的策略:
        // 删掉旧的,重新建个新的插到头部,逻辑异常清晰!
        if (mp.count(key))
        {
            remove(mp[key]);
            delete mp[key];
            mp[key] = NULL;
        }

        // 创建新节点,执行头插法 (hinert),并在 map 中记录位置
        Node *tmp = new Node(key, value);
        hinert(tmp);
        mp[key] = tmp;

        // 如果超出容量限制,启动 LRU 淘汰机制
        if (mp.size() > cap)
        {
            // 真实有效节点中,最久没使用的就是 tail 的前一个节点
            Node *todel = tail -> pre;
            remove(todel);           // 从链表中剥离
            mp.erase(todel -> key);  // 从哈希表中注销
            delete todel;            // 释放内存,防止内存泄漏
        }
    }

    // 辅助函数:将节点从双向链表中剥离
    void remove(Node *tmp)
    {
        tmp -> pre -> next = tmp -> next;
        tmp -> next -> pre = tmp -> pre;
    }

    // 辅助函数:头插法 (Head Insert),将节点插入到虚拟头节点之后
    void hinert(Node *tmp)
    {
        tmp -> next = head -> next;
        head -> next = tmp;
        tmp -> pre = head;
        tmp -> next -> pre = tmp;
    }

private:
    int cap;
    Node *head, *tail;
    unordered_map<int, Node*> mp; // Key 到 链表节点指针 的映射
};
相关推荐
用户8055336980318 小时前
不止三件套:QObject 属性系统全关键字与运行时反射!
c++·qt
To_OC1 天前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode
To_OC1 天前
LC 208 实现 Trie 前缀树:曾被名字劝退,写完发现是送分题
javascript·算法·leetcode
BadBadBad__AK1 天前
线段树维护区间 k 次方和
c++·数学·算法·stl
卷无止境2 天前
Eigen 库如何借助 OpenMP 加速计算
c++·后端
卷无止境2 天前
OpenMPI、MPICH 与 OpenMP:关系、核心概念与架构全解
c++·后端
To_OC2 天前
LC 994 腐烂的橘子:人人都说是 BFS 入门题,我却写了三遍才过
javascript·算法·leetcode
To_OC3 天前
LC 200 岛屿数量:经典 DFS 入门题,我第一次写居然连方向都搞错了
javascript·算法·leetcode
郝学胜_神的一滴3 天前
CMake 30:循环语法全解|foreach_while双循环精讲、迭代技巧与实战避坑指南
c++·cmake
To_OC3 天前
LC 128 最长连续序列:别上来就排序,O (n) 解法才是这题的灵魂
javascript·算法·leetcode