当前位置: 首页 > news >正文

力扣——146.LRU缓存

题目链接:

https://leetcode.cn/problems/lru-cache/solutions/259678/lruhuan-cun-ji-zhi-by-leetcode-solution/?envType=study-plan-v2&envId=top-100-liked

题目描述:

在这里插入图片描述

思路:

  1. 提到key-value一定有map;
  2. 要实现最近最少使用,最常见的一种思路就是使用flag,某一个数最近使用了,让其flag+1,这种方式比较麻烦,因为如果数据很多,每一个都需要flag,占用的空间变大;
  3. 另一种思路就是排序,如果这个数最近使用了,就让他排到前/后面,这就需要这些数可以随意的前后移动,可以使用链表;这里规定排在前面
  4. 如果超过容量,需要删除排在最后面的数并把新加的数放在最前面,
  5. 删除的操作比较麻烦,因为要找到要删除节点的前后节点,如果用单向链表,需要遍历才能找到他前面的,如果用双向链表就方便了
  6. 可以加入虚拟头和虚拟尾,这样就不需要对头结点和尾节点做特殊处理了

综上,需要使用map和双向链表完成

具体要怎么做:
数据存储方式:map里面存放key和node,所有的node组成一个双向链表,node是双向链表的节点,node里面存放他的前后节点,以及key、value这两个值,一般来说node里面只需要存放value就行,但是这里也要放key,因为后面会用到。

对于新增:要把key和node放入map里面;先看一下map里面有没有这个key,有的话就更新原有node的value值;没有的话就加到map里面,同时,node加入双向链表的最前面,新增后还要看链表的长度是否已经超过容量,超过的话就把最后一个node删除,注意也要把map里面的key删除,这个key在node里面有(这就是为什么node里面也要存key)。

对于获取:先看一下map里面有没有这个key,有的话返回这个node,并把这个node放到链表最前面

实现代码:

class LRUCache {

    class Dulinked{
        Dulinked prev;
        Dulinked next;
        int key;
        int value;
        public Dulinked(){};
        public Dulinked(int _key,int _value){
            key = _key;
            value = _value;
        }
    }


    private int size;
    private Map<Integer,Dulinked> table = new HashMap<Integer,Dulinked>();
    private int capacity;
    private Dulinked head,end;

    public LRUCache(int capacity) {
        this.size = 0;
        this.capacity = capacity;
        head = new Dulinked();
        end = new Dulinked();
        head.next = end;
        end.prev = head;
    }
    
    public int get(int key) {
        Dulinked node = table.get(key);
        if(node == null){
            return -1;
        }
        moveToHead(node);
        return node.value;

    }
    
    public void put(int key, int value) {
        Dulinked node = table.get(key);
        if(node == null){
            Dulinked newNode = new Dulinked(key,value);
            table.put(key,newNode);
            addToHead(newNode);
            size++;
            if(size > capacity){
                Dulinked end = removeEnd();
                table.remove(end.key);
                size--;
            }
        }
        else{
            node.value = value;
            moveToHead(node);
        }
    }

    private void removeNode(Dulinked node){
        node.prev.next = node.next;
        node.next.prev = node.prev;
    }

    private void addToHead(Dulinked node){
        head.next.prev = node;
        node.next = head.next;
        node.prev = head;
        head.next = node;

    }
    
    private void moveToHead(Dulinked node){
        removeNode(node);
        addToHead(node);
    }

    private Dulinked removeEnd(){
        Dulinked res = end.prev;
        removeNode(res);
        return res;
    }

}

/**
 * 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);
 */

相关文章:

  • 【AD】5-19 PCB阻焊开窗及异形开窗
  • 五、Ultra-Fast-Lane-Detection 训练数据集转换
  • 802.11标准
  • Java8计算集合属性的平均值
  • 一文了解大模型Function Calling
  • 成绩排序(结构体排序)
  • JVM内存结构笔记01-运行时数据区域
  • 3.14学习总结
  • RISC-V汇编学习(五)—— 汇编实战、GCC内联汇编(基于芯来平台)
  • 【训练细节解读】文本智能混合分块(Mixtures of Text Chunking,MoC)引领RAG进入多粒度感知智能分块阶段
  • 【乐企板式文件】关于乐企板式文件使用OFD模板解析的方式实现说明
  • AAAI2025 Accepted Papers(二)
  • AWS Bedrock全托管接入国产大模型DeepSeek-R1[内涵免费使用DeepSeek-R1满血版]
  • 【0x80070666】-已安装另一个版本...(Tableau 安装失败)
  • MFC中使用Create或CreateDialog创建对话框失败,GetLastError错误码为1813(找不到映像文件中指定的资源类型)
  • linux 命令 case
  • 力扣——合并K个排序链表
  • Ubuntu 18,04 LTS 通过APT安装mips64el的交叉编译器。
  • 平安养老险广西分公司2025年“3∙15”金融消费者权益教育宣传活动暨南湖公园健步行活动
  • uni-app+SpringBoot: 前端传参,后端如何接收参数
  • 国家统计局:中美大幅降低关税有利于双方贸易增长,也有利于世界经济复苏
  • 西浦国际教育创新论坛举行,聚焦AI时代教育本质的前沿探讨
  • 复旦大学艺术馆开馆:以当代视角再看文科文脉
  • 15年全程免费,内蒙古准格尔旗实现幼儿园到高中0学费
  • 气急败坏!20多名台湾艺人被台当局列为“重点核查对象”
  • 上海比常年平均时间提前12天入夏,明天最高气温可达33℃