hot100题解 —— 146.LRU缓存

一、题目

二、题解

1. 思路

双向链表 + 哈希表。

哈希表用于快速查找是否存在值,表中存放链表节点 - key;双向链表用于快速确定最近使用情况,链表节点存放key和value。

每次访问时,将节点移动到链表头部,当内部节点超出容量时,删掉尾部节点。

2. 题解

java 复制代码
class LRUCache {
    class DLinkedNode {
        int key;
        int value;
        DLinkedNode prev;
        DLinkedNode next;
        public DLinkedNode() {}
        public DLinkedNode(int _key, int _value) {key = _key; value = _value; }
    }
    // 双向链表存键值对,哈希表存节点+键
    private Map<Integer, DLinkedNode> cache = new HashMap<Integer, DLinkedNode>();
    private int size;
    private int Capacity;
    private DLinkedNode head, tail;

    public LRUCache(int capacity) {
        // 初始化值
        this.size = 0;
        this.Capacity = capacity;
        // 构建空的双向链表
        head = new DLinkedNode();
        tail = new DLinkedNode();
        head.next = tail;
        tail.prev = head;
    }
 
    public int get(int key) {
        // 查询key
        DLinkedNode findNode = cache.get(key);
        // 若无该节点
        if(findNode == null){
            return -1;
        } else {
            // 有key ------ 返回value
            moveToHead(findNode);
            return findNode.value;
        }
    }
    
    public void put(int key, int value) {
        DLinkedNode node = cache.get(key);
        
        if(node == null){
            // 如果不存在,先插入,再判断是否空间溢出
            DLinkedNode newNode = new DLinkedNode(key, value);
            cache.put(key, newNode);
            addToHead(newNode);
            size++;
            if(size > Capacity) {
                DLinkedNode tailNode = removeTailNode();
                cache.remove(tailNode.key);
                size--;
            }
        } else {
            // 如果存在,则更新value
            node.value = value;
            moveToHead(node);
        }
    }

    // 移除节点
    private void removeNode(DLinkedNode node){
        node.prev.next = node.next;
        node.next.prev = node.prev;
    } 

    // 在头部插入节点
    private void addToHead(DLinkedNode node){
        node.next = head.next;
        head.next.prev = node;
        head.next = node;
        node.prev = head;
    }

    // 将节点挪到头部
    private void moveToHead(DLinkedNode node){
        removeNode(node);
        addToHead(node);
    }

    // 删除尾部节点
    private DLinkedNode removeTailNode(){
        DLinkedNode res = tail.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);
 */
相关推荐
阿维的博客日记6 分钟前
Hippo4j 线程池监控平台部署手册
java·spring boot·后端
南境十里·墨染春水2 小时前
C++ 工厂模式:从入门到进阶,彻底掌握对象创建的艺术
开发语言·c++·算法
C+++Python2 小时前
详细介绍一下Java泛型的通配符
java·windows·python
JosieBook3 小时前
【数据库】时序预测能力的分级进化:TimechoAI如何让每一类用户都能精准预见未来
java·开发语言·数据库
加号33 小时前
【C#】 文件与目录管理:创建、删除操作的技术解析
开发语言·c#
diving deep4 小时前
脚本速览-python
开发语言·python
一生了无挂4 小时前
Java处理JSON技巧教学(从基础到高阶实战全覆盖)
java·开发语言·json
李白的天不白4 小时前
使用 SmartAdmin 进行前后端开发
java·前端
swordbob4 小时前
Spring 单例 Bean 是线程安全的吗?
java·开发语言
2601_951643775 小时前
Python第一,Java跌出前三,C语言杀回来了
java·c语言·python·编程语言排行·技术趋势