lc146LRU缓存——模仿LinkedHashMap

146. LRU 缓存 - 力扣(LeetCode)

法1:

调用java现有的LinkedHashMap的方法,但不太理解反正都不需要扩容,super(capacity, 1F, true);不行吗,干嘛还弄个装载因子0.75还中途扩容一次浪费时间。

复制代码
class LRUCache extends LinkedHashMap<Integer, Integer>{
    private int capacity;
    
    public LRUCache(int capacity) {
        super(capacity, 0.75F, true);
        this.capacity = capacity;
    }

    public int get(int key) {
        return super.getOrDefault(key, -1);
    }

    public void put(int key, int value) {
        super.put(key, value);
    }

    @Override
    protected boolean removeEldestEntry(Map.Entry<Integer, Integer> eldest) {
        return size() > capacity; 
    }
}

作者:力扣官方题解
链接:https://leetcode.cn/problems/lru-cache/solutions/259678/lruhuan-cun-ji-zhi-by-leetcode-solution/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

法2:

要实现O(1)的get和put,那得用哈希表但手撸一个哈希难度太大,应该不至于考那么过分,那就基于使用现有map的前提下实现LRU缓存。关键是要记录最近最少使用的是哪一个,那么需要每次将最新使用的放到"最后面",这样每次取"最前面"那个进行淘汰就是最近最少使用的。那么采用双向链表比较合适,队列,栈又不能从中间取元素;数组从中间移动到最前面,移动太多,时间复杂度高;单向链表的话需要知道被移动元素的前一个元素才方便移动,所以最后相比之下双向链表最合适。

双向链表放哪些内容呢,前后指针是要放的,value是要放的,key也需要因为当容量满了要删除map中head指向的节点,那么就要从这个双向链表节点中获取到key才行。

另外可以使用哑结点技巧,即head,tail都用一个不存value的空节点表示,也叫伪头结点,伪尾结点。这样有很多好处,1.当你尾插入时,正常不用哑结点要分3种情况讨论,(1)插入时,链表为空,head,tail都为空;(2)head=tail时即仅一个节点时(3)2个即以上节点时(正常情况时),

2,当删除链表中head时,也要分2种情况,(1)capacity为1时,head = tail;(2)正常情况;

当使用哑结点技巧后,尾插和删除head都只用1种情况处理,简化太多。

复制代码
class LRUCache {
    class DoubleLinkedList {
        int key; //不存key,到时候不好删除
        int val;
        DoubleLinkedList prev;
        DoubleLinkedList next;

        public DoubleLinkedList() {};
        public DoubleLinkedList(int key, int val) {
            this.key = key;
            this.val = val;
        }
    };

    private Map<Integer, DoubleLinkedList> map;
    private DoubleLinkedList head;
    private DoubleLinkedList tail;
    private int capacity;
    private int size;

    public LRUCache(int capacity) {
        map = new HashMap<Integer, DoubleLinkedList>(capacity, 1F);
        //哑结点
        head = new DoubleLinkedList();
        tail = new DoubleLinkedList();
        head.next = tail;
        tail.prev = head;
        this.capacity = capacity;
        size = 0;
    }
    
    public int get(int key) {
        if(!map.containsKey(key)) {
            return -1;
        }

        //含有,则1.更新到链表末尾,2.返回
        DoubleLinkedList node = map.get(key);
        //更新到链表尾部
        move2Tail(node);

        return node.val;
    }
    
    public void put(int key, int value) {
        //1.之前就存在
        if(map.containsKey(key)) {
            DoubleLinkedList node = map.get(key);
            node.val = value;
            //更新最新使用
            move2Tail(node);
            return;
        } else if(size < capacity) {
            //2.容量没满 新添加
            add(key, value);
            size++;
            return;
        } 
        // 3.满了需要替换
        //3.1删除最久没使用的
        //tail末尾也搞个哑结点,不然capacity为1时,删除时还要考虑head.next.next为空的情况
        map.remove(head.next.key);
        deleteNode(head.next);
        // 3.2新添加进map
        add(key, value);
    }

    //更新最新使用,移到链表尾部
    public void move2Tail(DoubleLinkedList node) {
        //正常分3种情况,1.是head,2.在中间,3.本就在tail
        //利用头部和尾部哑结点,头,尾结点先用一个没有实际意义的节点,就可以1,2,3情况全合并了

        // 1.原链表中先删除节点
        deleteNode(node);
        // 2.插入到末尾
        insert2Tail(node);
    }

    //从链表中删除节点
    public void deleteNode(DoubleLinkedList node) {
        node.prev.next = node.next;
        node.next.prev = node.prev;
    }

    //新增map节点
    public void add(int key, int value) {
        DoubleLinkedList newNode = new DoubleLinkedList(key, value);
        //用了尾部哑结点,就不管是不是第一次插入都同样处理了
        map.put(key, newNode);
        insert2Tail(newNode);
    }

    //插入到尾部
    public void insert2Tail(DoubleLinkedList node) {
        node.prev = tail.prev;
        tail.prev.next = node;
        node.next = tail;
        tail.prev = 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);
 */
相关推荐
SPC的存折4 小时前
1、Redis数据库基础
linux·运维·服务器·数据库·redis·缓存
身如柳絮随风扬11 小时前
Redis如何实现高效插入大量数据
数据库·redis·缓存
予早12 小时前
Redis 设置库的数量
数据库·redis·缓存
黑金IT12 小时前
vLLM本地缓存实战,重复提交直接复用不浪费算力
人工智能·缓存
Rick199314 小时前
Redis查询为什么快
数据库·redis·缓存
Rick199315 小时前
Redis 底层架构图
数据库·redis·缓存
Arva .16 小时前
Redis 数据类型
数据库·redis·缓存
笑我归无处17 小时前
Redis和数据库的数据一致性问题研究
数据库·redis·缓存
小红的布丁17 小时前
操作系统与高性能 IO:零拷贝、一次读 IO、CPU 缓存与伪共享
缓存
SPC的存折18 小时前
(自用)LNMP-Redis-Discuz5.0部署指南-openEuler24.03-测试环境
linux·运维·服务器·数据库·redis·缓存