力扣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);
 */
相关推荐
凌肖战1 小时前
力扣网C语言编程题:在数组中查找目标值位置之二分查找法
c语言·算法·leetcode
GEEK零零七3 小时前
Leetcode 1103. 分糖果 II
数学·算法·leetcode·等差数列
苦夏木禾5 小时前
js请求避免缓存的三种方式
开发语言·javascript·缓存
重庆小透明5 小时前
力扣刷题记录【1】146.LRU缓存
java·后端·学习·算法·leetcode·缓存
desssq5 小时前
力扣:70. 爬楼梯
算法·leetcode·职场和发展
岁忧6 小时前
(LeetCode 面试经典 150 题 ) 58. 最后一个单词的长度 (字符串)
java·c++·算法·leetcode·面试·go
Java初学者小白6 小时前
秋招Day14 - Redis - 应用
java·数据库·redis·缓存
奈斯ing7 小时前
【Redis篇】数据库架构演进中Redis缓存的技术必然性—高并发场景下穿透、击穿、雪崩的体系化解决方案
运维·redis·缓存·数据库架构
??tobenewyorker9 小时前
力扣打卡第二十一天 中后遍历+中前遍历 构造二叉树
数据结构·c++·算法·leetcode
凌肖战12 小时前
力扣网C语言编程题:快慢指针来解决 “寻找重复数”
c语言·算法·leetcode