15 LRU缓存

146. LRU 缓存 - 力扣(LeetCode)

这道题采用双向链表加哈希表;

哈希表是为了随机访问,双向链表是为了能够确定位置

这里面注意的是我们需要一个哨兵节点来辅助,需要让哨兵节点的prev.next以及next.next指向自己,即这里是一个双向循环链表,并且我们每次头插节点的时候都是头插在哨兵节点之后

复制代码
class LRUCache {
    //这里put和get想实现O1那么就需要使用哈希表,但是哈希表是没有位置观念的
    //比如我们想删除最久插入的元素,那么只能一个个遍历,而有位置观念的就是数组和链表了
    //但是我们每次插入元素的时候都要放到最前面也就是头插,如果用数组的话,那么元素需要
    //整体移动,所以采用双向链表

    static class Node{
        int key;
        int val;
        Node prev;
        Node next;
        Node(int k ,int v){
            key = k;
            val = v;
        }
    }
    private int capacity;
    private Map<Integer,Node> map = new HashMap<>();
    //设置一个哨兵节点
    private Node head  = new Node(0,0);
 
 //构造初始化
    public LRUCache(int capacity) {
        this.capacity = capacity;
        head.next = head;
        head.prev = head;
    }
    
    public int get(int key) {
        //获取最新节点的值
        Node node = getNode(key);
        return node == null ? -1 : node.val;
        
    }
    
    public void put(int key, int value) {
        Node node = getNode(key);
        if(node != null){
            map.put(key,node);
            node.val = value;
            return;
        } 
        //不存在插入
        Node newNode = new Node(key,value);
        addLeft(newNode);
        map.put(key,newNode);
        if(map.size() > capacity) {
            Node last = head.prev;
            remove(last);
            map.remove(last.key);
        }
        
    }
    private Node getNode(int key){
        Node node = map.get(key);
        if(node == null) return null;
        remove(node);
        addLeft(node);
        return node;
    }
    private void remove(Node node){
        node.next.prev = node.prev;
        node.prev.next = node.next;
    }
    private void addLeft(Node node){
        //插到烧饼节点后面
        node.prev = head;
        node.next = head.next;
        head.next.prev = node;
        head.next = node;
        // node.prev.next = node;
        // node.next.prev = node;
    }
}
相关推荐
董董灿是个攻城狮17 小时前
AI视觉连载8:传统 CV 之边缘检测
算法
AI软著研究员1 天前
程序员必看:软著不是“面子工程”,是代码的“法律保险”
算法
FunnySaltyFish1 天前
什么?Compose 把 GapBuffer 换成了 LinkBuffer?
算法·kotlin·android jetpack
颜酱1 天前
理解二叉树最近公共祖先(LCA):从基础到变种解析
javascript·后端·算法
地平线开发者2 天前
SparseDrive 模型导出与性能优化实战
算法·自动驾驶
董董灿是个攻城狮2 天前
大模型连载2:初步认识 tokenizer 的过程
算法
地平线开发者2 天前
地平线 VP 接口工程实践(一):hbVPRoiResize 接口功能、使用约束与典型问题总结
算法·自动驾驶
罗西的思考2 天前
AI Agent框架探秘:拆解 OpenHands(10)--- Runtime
人工智能·算法·机器学习
HXhlx2 天前
CART决策树基本原理
算法·机器学习
Wect2 天前
LeetCode 210. 课程表 II 题解:Kahn算法+DFS 双解法精讲
前端·算法·typescript