手撕LRU缓存——LinkedHashMap简易源码

题目链接:https://leetcode.cn/problems/lru-cache/description/?envType=study-plan-v2&envId=top-100-liked

原理非常简单,一个双端链表配上一个hash表。

首先我们要知道什么是LRU就是最小使用淘汰。怎么淘汰,链表尾部就是最不常用的直接删除,最近使用过的就移动到链表头部。要查找就利用hash表进行映射查找。

代码:

java 复制代码
class LRUCache {
    //需要定义的双端链表节点
    //哈希表
    class Node{
        int key;
        int value;
        Node prev;
        Node next;
        public Node() {}
        public Node(int _key,int _value) {
            key = _key;
            value = _value;
        }
    }
    private Map<Integer,Node > cache = new HashMap<>();
    private int size;
    private int capacity;
    private Node head,tail;
    public LRUCache(int capacity) {
        this.size = 0;
        this.capacity = capacity;
        head = new Node();
        tail = new Node();
        head.next = tail;
        tail.prev = head;
    }
    
    public int get(int key) {
        Node node = cache.get(key);
        if(node == null) 
            return -1;
        moveToHead(node);
        return node.value;
    }
    
    public void put(int key, int value) {
        //如果存在
        if(cache.containsKey(key)){
            Node t = cache.get(key);
            t.value = value;
            moveToHead(t);
        }else{
            //如果不存在
            size++;
            if(size>capacity){
                cache.remove(tail.prev.key);
                //System.out.println(tail.prev.key);
                moveTail();
                size--;
            }
            Node node = new Node(key,value);
            Node temp = head.next;
            head.next = node;
            node.prev = head;
            temp.prev = node;
            node.next = temp;
            cache.put(key,node);
        }
    }

    public void moveToHead(Node node){
        //本身删除了
        Node qian = node.prev;
        qian.next = node.next;
        node.next.prev = qian;
        //再插到头部
        Node temp = head.next;
        head.next = node;
        node.prev = head;
        temp.prev = node;
        node.next = temp;
    }

    public void moveTail(){
        Node temp = tail.prev.prev;
        temp.next = tail;
        tail.prev = 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);
 */
相关推荐
踩坑小念3 小时前
秒杀场景下如何处理redis扣除状态不一致问题
数据库·redis·分布式·缓存·秒杀
J_liaty5 小时前
Redis 超详细入门教程:从零基础到实战精通
数据库·redis·缓存
難釋懷7 小时前
优惠卷秒杀集群环境下的并发问题
redis·缓存
可涵不会debug9 小时前
Redis魔法学院——第四课:哈希(Hash)深度解析:Field-Value 层级结构、原子性操作与内部编码优化
数据库·redis·算法·缓存·哈希算法
fengxin_rou9 小时前
【黑马点评实战篇|第一篇:基于Redis实现登录】
java·开发语言·数据库·redis·缓存
我待_JAVA_如初恋9 小时前
Redis常用的数据类型之String
数据库·redis·缓存
ALex_zry10 小时前
分布式缓存与微服务架构的集成
分布式·缓存·架构
ALex_zry11 小时前
分布式缓存安全最佳实践
分布式·安全·缓存
Anastasiozzzz11 小时前
数据库与缓存的一致性之间的终极博弈!
缓存
Demon_Hao11 小时前
JAVA缓存的使用RedisCache、LocalCache、复合缓存
java·开发语言·缓存