【深究系列】LRU算法设计

【深究系列】LRU算法设计

一、LeetCode算法题目

https://leetcode-cn.com/problems/lru-cache/

image-20220215223937136

所谓缓存,必须要有读写两个操作,按照命中率的思路考虑,写操作+读操作的时间复杂度都需要O(1)

特性要求:

必须要有顺序之分,以区分最近使用的和很久没有使用的数据排序

写和读操作一次搞定。

如果容量满了要删除最不常用的数据,每次新访问还要把新的数据插入到队头(按照业务你自己设定左右哪一边是队头)

查找快、插入块、删除快,且还需要先后排序?——什么样的数据结构可以满足这个问题?

二、LRU算法核心

LRU的算法核心是哈希链表

本质就是HashMap + DoubleLinkedList,哈希表+双向链表的结合体。

查找用哈希,增删用链表。

三、巧用LinkedHashMap完成

1、参考LinkedHashMap

class LRUCache extends LinkedHashMap<Integer, Integer>{
    private int capacity;

    public LRUCache(int capacity) {
        //true,false按照读取顺序或者插入顺序
        super(capacity, 0.75f, true);
        this.capacity = capacity;
    }
    
    public int get(int key) {
        //返回key,或者-1
        return super.getOrDefault(key, -1);
    }
    
    public void put(int key, int value) {
        super.put(key, value);
    }

    //删除最老的节点的条件
    protected boolean removeEldestEntry(Map.Entry<Integer, Integer> eldest) {
        return super.size() > capacity;
    }
}

四、手写一个LRU

核心思想

完整代码

image-20220216124048260

import java.util.HashMap;
import java.util.Map;

public class LRUDemo2 {
    //map负责查找,构建一个虚拟的双向链表,它里面安装的就是一个个Node结点
    //作为数据载体

    //1.构造一个Node节点,作为数据载体
    class Node<K, V>{
        K key;
        V value;
        Node<K, V> prev;
        Node<K, V> next;

        public Node() {
            this.prev = this.next = null;
        }

        public Node(K key, V value) {
            this.key = key;
            this.value = value;
            this.prev = this.next = null;
        }
    }

    //2. 构建一个虚拟的双向链表,里面安放的就是我们的Node
    class DoubleLinkedList<K, V> {
        Node<K, V> head;
        Node<K, V> tail;

        //2.1构造方法
        public DoubleLinkedList() {
            head = new Node<>();
            tail = new Node<>();
            head.next = tail;
            tail.prev = head;
        }

        //2.2添加到头
        public void addHead(Node<K, V> node) {
            node.next = head.next;
            node.prev = head;
            head.next.prev = node;
            head.next = node;
        }

        //2.3 删除节点
        public void removeNode(Node<K, V> node) {
            node.next.prev = node.prev;
            node.prev.next = node.next;
            node.prev = null;
            node.next = null;
        }

        //2.4 获得最后一个节点
        public Node getLast() {
            return tail.prev;
        }
    }

    public int cacheSize;
    Map<Integer, Node<Integer, Integer>> map;
    DoubleLinkedList<Integer, Integer> doubleLinkedList;

    public LRUDemo2(int cacheSize) {
        this.cacheSize = cacheSize;//坑位
        map = new HashMap<>();//负责查找
        doubleLinkedList = new DoubleLinkedList<>();
    }

    public int get(int key) {
        if (!map.containsKey(key)) {
            return -1;
        }

        Node<Integer, Integer> node = map.get(key);
        doubleLinkedList.removeNode(node);//把这个结点删掉
        doubleLinkedList.addHead(node);

        return node.value;
    }

    //saveOrUpdate method
    public void put(int key, int value) {
        if (map.containsKey(key)) {
            Node<Integer, Integer> node = map.get(key);
            node.value = value;
            map.put(key, node);

            doubleLinkedList.removeNode(node);
            doubleLinkedList.addHead(node);
        } else {
            if (map.size() == cacheSize) {//坑位满了
                Node<Integer,Integer> lastNode = doubleLinkedList.getLast();
                map.remove(lastNode.key);
                doubleLinkedList.removeNode(lastNode);
            }
            //才是新增一个
            Node<Integer, Integer> newNode = new Node<>(key, value);
            map.put(key, newNode);
            doubleLinkedList.addHead(newNode);
        }
    }
}
posted @ 2022-02-16 12:47  DarkerG  阅读(67)  评论(0编辑  收藏  举报