leetcode刷题日志-146LRU缓存

思路:使用hashmap储存key,vaule,使用双向链表以快速查到尾结点(待逐出的节点),链表的题一定要在纸上画一下,不然连着连着就不知道连在哪里去了

java 复制代码
class LRUCache {
    public class ListNode {
      int key;
      int value;
      ListNode next;
      ListNode pre;
      ListNode() {}
      ListNode(int key , int value) { this.key = key; this.value = value;}
      ListNode(int kye, int value, ListNode next) { this.key = key;this.value = value;this.next = next;}
      ListNode(int kye, int value, ListNode next,ListNode pre) { this.key = key;this.value = value;this.next = next;this.pre = pre;}
  }
    ListNode head;//头节点
    ListNode tail;//指向尾结点的前一个节点,方便逐出最久未使用关键字
    int capacity; //储存容量
    int cur_capacity;//储存当前容量
    Map<Integer,ListNode> map;//储存节点
    
    public LRUCache(int capacity) {
        this.head = new ListNode(0,0,null,null);
        this.capacity = capacity;
        this.cur_capacity = 0;
        this.map = new HashMap<>();
        this.tail = new ListNode(0,0,null,head);
        this.head.next = tail;
    }
    
    public int get(int key) {
        if(map.containsKey(key))  //最近使用到的key,将位置提前
        {
            if(map.get(key).next != null)
            {
                map.get(key).next.pre = map.get(key).pre;
                map.get(key).pre.next = map.get(key).next;
            }
            map.get(key).next = head.next;
            map.get(key).next.pre = map.get(key);
            map.get(key).pre = head;
            head.next = map.get(key);
            return map.get(key).value;
        }
        else
            return -1;
    }
    
    public void put(int key, int value) {
        if(map.containsKey(key)) //存在,将位置提前
        {
            map.get(key).value = value;
            if(map.get(key).next != null)
            {
                map.get(key).next.pre = map.get(key).pre;
                map.get(key).pre.next = map.get(key).next;
            }
            map.get(key).next = head.next;
            map.get(key).next.pre = map.get(key);
            map.get(key).pre = head;
            head.next = map.get(key);
        }
        else{//不存在
            if(this.cur_capacity < this.capacity) //容量足够,直接添加到头
            {
                ListNode temp = new ListNode();
                temp.key = key;
                temp.value = value;
                temp.next = head.next;
                if(temp.next != null)
                temp.next.pre = temp;
                temp.pre = head;
                head.next = temp;
                cur_capacity++;
                map.put(key,temp);
            }
            else//容量不够,移出尾结点,添加新节点到头
            {

                ListNode tail_pre = tail.pre;
                map.remove(tail_pre.key);
                tail.pre.pre.next = tail;
                tail.pre.next = null;
                tail.pre = tail.pre.pre;
                tail_pre.pre = null;
                ListNode temp = new ListNode();
                temp.key = key;
                temp.value = value;
                temp.next = head.next;
                temp.next.pre = temp;
                temp.pre = head;
                head.next = temp;
                map.put(key,temp);
            }
        }
    }
}

/**
 * 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);
 */
相关推荐
power 雀儿5 分钟前
Scaled Dot-Product Attention 分数计算 C++
算法
琹箐35 分钟前
最大堆和最小堆 实现思路
java·开发语言·算法
renhongxia11 小时前
如何基于知识图谱进行故障原因、事故原因推理,需要用到哪些算法
人工智能·深度学习·算法·机器学习·自然语言处理·transformer·知识图谱
坚持就完事了1 小时前
数据结构之树(Java实现)
java·算法
算法备案代理1 小时前
大模型备案与算法备案,企业该如何选择?
人工智能·算法·大模型·算法备案
赛姐在努力.2 小时前
【拓扑排序】-- 算法原理讲解,及实现拓扑排序,附赠热门例题
java·算法·图论
野犬寒鸦3 小时前
从零起步学习并发编程 || 第六章:ReentrantLock与synchronized 的辨析及运用
java·服务器·数据库·后端·学习·算法
霖霖总总3 小时前
[小技巧66]当自增主键耗尽:MySQL 主键溢出问题深度解析与雪花算法替代方案
mysql·算法
rainbow68893 小时前
深入解析C++STL:map与set底层奥秘
java·数据结构·算法
wangjialelele3 小时前
平衡二叉搜索树:AVL树和红黑树
java·c语言·开发语言·数据结构·c++·算法·深度优先