力扣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);
 */
相关推荐
yzlAurora37 分钟前
删除链表倒数第N个节点
数据结构·链表
鱼儿也有烦恼1 小时前
Redis最新入门教程
数据库·redis·缓存
进击的小白菜1 小时前
如何高效实现「LeetCode25. K 个一组翻转链表」?Java 详细解决方案
java·数据结构·leetcode·链表
悻运2 小时前
Spark缓存--persist方法
大数据·缓存·spark
勤奋的小王同学~3 小时前
计算机网络常识:缓存、长短连接 网络初探、URL、客户端与服务端、域名操作 tcp 三次握手 四次挥手
缓存
天上掉下来个程小白3 小时前
缓存菜品-04.功能测试
java·spring boot·缓存·微信小程序·需求分析·苍穹外卖
我是一只鱼02233 小时前
LeetCode算法题 (反转链表)Day17!!!C/C++
数据结构·c++·算法·leetcode·链表
LuckyLay3 小时前
LeetCode算法题(Go语言实现)_62
算法·leetcode·职场和发展
元亓亓亓4 小时前
LeetCode热题100--54.螺旋矩阵--中等
算法·leetcode·矩阵
秀才恶霸4 小时前
04-redis
数据库·redis·缓存