Java实现LRU算法

1、内存空间有限,当缓存满的时候,如何淘汰缓存?

FIFO(First In First Out)先进先出
LFU(Least Frequently Used)最不经常使用
LRU(Least Recently Used)最近最少使用

2、实现LRU demo

1、使用Java容器LinkedHashMap

LinkedHashMap本身就具有LRU算法的特性

    class LRUCache {
        private Map<Integer, Integer> cacheMap = null;

        public LRUCache(int capacity) {
            // 参数设置true,当removeEldestEntry()返回true,则删除最旧的数据
            cacheMap = new LinkedHashMap<Integer, Integer>(capacity,0.75F,true){
                @Override
                protected boolean removeEldestEntry(Map.Entry<Integer, Integer> eldest) {
                    return size() > capacity;
                }
            };
        }

        public int get(int key) {
            return cacheMap.getOrDefault(key, -1);
        }

        public void put(int key, int value) {
            cacheMap.put(key, value);
        }
    }

2、哈希表(HashMap)+双向链表

  • 维护一个双向链表,靠近链表尾部的结点是越早访问的,靠近头部的节点是最近访问的。
  • 如果此数据之前已经被缓存在链表中了,我们遍历得到这个数据对应的结点,并将其从原来的位置删除,然后再插入到链表的头部。
  • 如果此数据没有在缓存链表中,又可以分为两种情况:
    如果此时缓存未满,则将此结点直接插入到链表的头部
    如果此时缓存已满,则链表尾结点删除,将新的数据结点插入链表的头部。

在这里插入图片描述

class LRUCache2 {
        // 当前缓存容量
        private int size;
        // 限制最大缓存容量
        private int capacity;
        // 定义伪头尾节点
        private Node head;
        private Node tail;
        // 定义HashMap存储数据
        private Map<Integer, Node> cache = new HashMap();

        // 初始化操作
        public LRUCache2(int capacity) {
            size = 0;
            this.capacity = capacity;
            // 初始化头尾节点
            head = new Node();
            tail = new Node();
            // 让头尾节点相联
            head.next = tail;
            tail.pre = head;
        }

        public int get(int key) {
            Node node = cache.get(key);
            // 不存在返回-1
            if (null == node) {
                return -1;
            }
            // 存在返回值,并且将当前节点移动到头
            moveNodeHead(node);
            return node.value;
        }

        public void put(int key, int value) {
            Node node = cache.get(key);
            // 不存在则插入,插入后判断当前容量是否大于限制最大容量
            if (null == node) {
                Node newNode = new Node(key, value);
                cache.put(key, newNode);
                // 放入链表头部
                addNodeHead(newNode);
                size++;
                if (size > capacity) {
                    // 删除尾结点
                    Node tail = removeNodeTail();
                    cache.remove(tail.key);
                }
            } else {
                // 存在则覆盖value,并且将当前节点移动到头
                node.value = value;
                moveNodeHead(node);
            }
        }

        // 放入链表的头部
        private void addNodeHead(Node node) {
            // 当前头节点的下一个节点的pre和当前节点连接
            head.next.pre = node;
            //
            node.next = head.next;
            //
            head.next = node;
            node.pre = head;
        }

        // 将节点移动到链表的头部
        private void moveNodeHead(Node node) {
            node.pre.next = node.next;
            node.next.pre = node.pre;
            addNodeHead(node);
        }

        // 删除链表的尾结点
        private Node removeNodeTail() {
            Node tailNode = tail.pre;
            tailNode.pre.next = tailNode.next;
            tailNode.next.pre = tailNode.pre;
            return tailNode;
        }

        // 定义一个双向链表,实际的缓存
        class Node {
            private int key;
            private int value;
            private Node pre;
            private Node next;
            public Node() {
            }
            public Node(int key, int value) {
                this.key = key;
                this.value = value;
            }
        }
    }
  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
LRU算法(最近最少使用)是一种常见的缓存淘汰算法,它根据数据最近被访问的时间来选择淘汰哪些数据。下面是一个简单的Java实现LRU算法的例子: ```java import java.util.LinkedHashMap; import java.util.Map; public class LRUCache<K, V> extends LinkedHashMap<K, V> { private final int MAX_CACHE_SIZE; public LRUCache(int cacheSize) { super((int) Math.ceil(cacheSize / 0.75) + 1, 0.75f, true); MAX_CACHE_SIZE = cacheSize; } @Override protected boolean removeEldestEntry(Map.Entry eldest) { return size() > MAX_CACHE_SIZE; } public static void main(String[] args) { LRUCache<String, Integer> cache = new LRUCache<>(3); cache.put("key1", 1); cache.put("key2", 2); cache.put("key3", 3); System.out.println(cache.keySet()); cache.put("key4", 4); System.out.println(cache.keySet()); cache.put("key3", 3); System.out.println(cache.keySet()); cache.put("key5", 5); System.out.println(cache.keySet()); } } ``` 在这个例子中,我们使用了Java的LinkedHashMap类,它可以按照插入顺序或访问顺序进行遍历。我们继承这个类,并重写了removeEldestEntry方法,当缓存的大小超过最大值时,自动删除最近最少使用的数据。我们还可以使用put方法向缓存中添加数据,使用get方法从缓存中获取数据。 在main方法中,我们创建了一个缓存大小为3的LRUCache对象,并向其中添加了4个键值对。在第一次打印缓存的键集合时,输出结果为[key1, key2, key3]。当添加第4个键值对时,由于缓存大小已经到达了最大值,所以自动删除最近最少使用的key1,输出结果为[key2, key3, key4]。当再次使用key3时,它将变成最新的,输出结果为[key2, key4, key3]。当添加第5个键值对时,由于缓存大小已经到达了最大值,所以自动删除最近最少使用的key2,输出结果为[key4, key3, key5]。 这个例子中使用了泛型,可以存储任何类型的数据。如果要存储其他类型的数据,只需要将K和V的类型改为相应的类型即可。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

李_杰

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值