力扣HOT100之链表:146. LRU 缓存

这道题从来没做过,完全不知道该怎么写,直接去看视频了,感觉这个视频讲解的挺好的。

这道题主要是需要自己额外定义数据结构和函数,需要定义节点结构体,用于存放键值对,每个节点都有前后指针,所以这道题是采用哈希表+双向链表的做法来做的。这道题添加和删除节点的逻辑都很好理解,最难想到的就是当插入节点,但缓存已满时,如何找到最久未使用的节点,这个实现起来不难,每一次插入节点都从头部插入,不常使用的节点总是在链表的最末端,所以我们只需要将末端的节点删除即可。

cpp 复制代码
//定义节点
struct Node{
    int key, val;
    Node *pre, *next;
    Node() : key(0), val(0), pre(nullptr), next(nullptr){}
    Node(int _key, int _val) : key(_key), val(_val), pre(nullptr), next(nullptr){}
};

class LRUCache {
public:
    Node *head, *tail;   //双向链表的头节点和尾节点
    unordered_map<int, Node*> hash;    //内部维护一个哈希表
    int capacity, size;   //容量和当前元素个数
    LRUCache(int _capacity) {
        capacity = _capacity;
        size = 0;
        head = new Node();
        tail = new Node();
        head -> next = tail;
        tail -> pre = head;
    }
    
    int get(int key) {
        if(!hash.count(key))
            return -1;
        Node* node = hash[key];
        removeNode(node);
        addNodeHead(node);
        return node -> val;
    }
    
    void put(int key, int value) {
        if(hash.count(key)){   //该键已经存在
            Node* node = hash[key];
            node -> val = value;
            removeNode(node);
            addNodeHead(node);
        }
        else{   //插入的为新键
            if(size >= capacity){  //已经达到最大容量
                Node* removed = tail -> pre;  //删除双向链表中的最后一个节点
                hash.erase(removed -> key);  //及时从哈希表中删除不活跃节点对应的键
                removeNode(removed);
                size--;
            }
            Node *node = new Node(key, value);
            addNodeHead(node);
            hash[key] = node;
            size++;
        }
    }
    //自定义删除节点函数
    void removeNode(Node *node){
        node -> pre -> next = node -> next;
        node -> next -> pre = node -> pre;
    }
    //自定义添加节点函数
    void addNodeHead(Node *node){
        node -> pre = head;
        node -> next = head -> next;
        head -> next -> pre = node;
        head -> 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);
 */
相关推荐
卧室小白5 分钟前
Redis-哨兵模式
数据库·redis·缓存
leoufung18 分钟前
LeetCode 76:Minimum Window Substring 题解与滑动窗口思维详解
算法·leetcode·职场和发展
卧室小白31 分钟前
redis-配置
数据库·redis·缓存
风筝在晴天搁浅2 小时前
LeetCode 92.反转链表Ⅱ
算法·leetcode·链表
WL_Aurora3 小时前
Python 算法基础篇之链表
python·算法·链表
普贤莲花5 小时前
【2026年第18周---写于20260501】---舍得
程序人生·算法·leetcode
m0_629494736 小时前
LeetCode 热题 100-----16.除了自身以外数组的乘积
数据结构·算法·leetcode
Lyyaoo.6 小时前
缓存更新策略
缓存
We་ct6 小时前
LeetCode 97. 交错字符串:动态规划详解
前端·算法·leetcode·typescript·动态规划
无敌昊哥战神6 小时前
【LeetCode 37】解数独 (Sudoku Solver) —— 回溯法详解 (Python/C/C++)
c语言·c++·python·算法·leetcode