Java简易LRU缓存实现(开发技巧)

威哥说:
缓存与APP应用的关系,好比西门庆与色,根本就离不开,而且地位最重,所以,小白们想成为老鸟,对待缓存之情必需要深入,以下文章给你介绍了Java如果实现LRU(Least Recently Used)算法,即最近最少使用算法,这个算法在Android开发中被用于图片,内容的缓存,我在《Android核心技术(下)》课程中就详细讲解了LRU的全部实现,有兴趣的朋友可以登录扣丁学堂学习。

背景
LinkedHashMap继承自HashMap,内部提供了一个removeEldestEntry方法,该方法正是实现LRU策略的关键所在,且HashMap内部专门为LinkedHashMap提供了3个专用回调方法,afterNodeAccess、afterNodeInsertion、afterNodeRemoval,这3个方法的字面意思非常容易理解,就是节点访问后、节点插入后、节点删除后分别执行的行为。基于以上行为LinkedHashMap就可以实现一个LRUCache的功能了。

关于LinkedHashMap的eldest:eldest字面意思为最老的,LinkedHashMap中有个叫做accessOrder的字段,当accessOrder为true时表示LinkedHashMap内部节点按照访问次数排序,最老的节点也就是访问最少的节点。当accessOrder为false时表示LinkedHashMap内部节点按照插入顺序排序,最老的节点也就是最早插入的节点,该值默认为false。

实现
自己实现LRUCache只需覆盖removeEldestEntry这个方法即可,代码如下

private static class LRUCache<K, V> extends LinkedHashMap<K, V>
    {
        private static final long serialVersionUID = -9111855653176630846L;
        private static int MAX_ELEMENTS;
        public LRUCache(int initCap, int maxSize) throws IllegalArgumentException
        {
            super(initCap, 0.75f, true);
            if (maxSize < 0)
                throw new IllegalArgumentException();
            MAX_ELEMENTS = maxSize;
        }
        @Override
        protected boolean removeEldestEntry(Map.Entry<K, V> eldest)
        {
            return size() > MAX_ELEMENTS;
        }
    }

以上代码需要一个MAX_ELEMENTS变量限制最大存储节点个数,插入节点时判断 如果当前节点个数已经超过了这个值则会根据LRU策略将访问最少的那个节点删除,这里需要注意,默认LinkedHashMap保证的是插入顺序,也就是节点按照插入先后来排序的,所以就算删除也是删除最先插入的节点,但是我们在构造函数中传入了一个true,这个参数决定了LinkedHashMap内部的节点按照什么方式排序,参数为true时说明内部节点按照最近访问的时间排序,为false时说明按照插入顺序排序。至此已完成了一个简易的LRUCache实现。

注意
由于LinkedHahsMap本身实现不是线程安全的,也就是说这个LRUCache也不是线程安全的,如果想要能多线程访问的话,可以这样使用它:LRUCache cache = Collections.synchronizedMap(new LRUCache(10, 10))。这样cache就可以在多线程下执行get\put等操作了,但是,用这种方式得到的cache在多线程遍历时还是不安全的。所以不能在多线程下遍历cache,官方文档也建议在遍历synchronizedmap时使用map本身做同步。

时间: 2024-10-12 16:35:39

Java简易LRU缓存实现(开发技巧)的相关文章

Java简易LRU缓存实现

背景 LinkedHashMap继承自HashMap,内部提供了一个removeEldestEntry方法,该方法正是实现LRU策略的关键所在,且HashMap内部专门为LinkedHashMap提供了3个专用回调方法,afterNodeAccess.afterNodeInsertion.afterNodeRemoval,这3个方法的字面意思非常容易理解,就是节点访问后.节点插入后.节点删除后分别执行的行为.基于以上行为LinkedHashMap就可以实现一个LRUCache的功能了. 实现 自

简易LRU缓存淘汰模拟

模拟采用链表保存缓存数据,功能如下 1. 新数据插入到链表头部: 2. 每当缓存命中(即缓存数据被访问),则将数据移到链表头部: 3. 当链表满的时候,将链表尾部的数据丢弃. 假设访问的数据为 Person public class LRUCache{ private static int count=0; private int capacity; private static class Node{ private static Node head; private static Node

LRU缓存介绍与实现 (Java)

引子: 我们平时总会有一个电话本记录所有朋友的电话,但是,如果有朋友经常联系,那些朋友的电话号码不用翻电话本我们也能记住,但是,如果长时间没有联系 了,要再次联系那位朋友的时候,我们又不得不求助电话本,但是,通过电话本查找还是很费时间的.但是,我们大脑能够记住的东西是一定的,我们只能记住自己 最熟悉的,而长时间不熟悉的自然就忘记了. 其实,计算机也用到了同样的一个概念,我们用缓存来存放以前读取的数据,而不是直接丢掉,这样,再次读取的时候,可以直接在缓存里面取,而不用再重 新查找一遍,这样系统的反

8.16 lru缓存java版

lru详细介绍及简单代码实现: http://blog.csdn.net/beiyetengqing/article/details/7855933 以下是本人的加强的lru缓存类,增加单例获取.缓存超时机制和修复一个clear()的bug package com.george.xblog.utils; import java.util.Hashtable; import java.util.Map; /** * Created by george.yang on 15/8/15. */ pub

LRU缓存实现(Java)

LRU Cache的LinkedHashMap实现 LRU Cache的链表+HashMap实现 LinkedHashMap的FIFO实现 调用示例 LRU是Least Recently Used 的缩写,翻译过来就是"最近最少使用",LRU缓存就是使用这种原理实现,简单的说就是缓存一定量的数据,当超过设定的阈值时就把一些过期的数据删除掉,比如我们缓存10000条数据,当数据小于10000时可以随意添加,当超过10000时就需要把新的数据添加进来,同时要把过期数据删除,以确保我们最大缓

转: LRU缓存介绍与实现 (Java)

引子: 我们平时总会有一个电话本记录所有朋友的电话,但是,如果有朋友经常联系,那些朋友的电话号码不用翻电话本我们也能记住,但是,如果长时间没有联系了,要再次联系那位朋友的时候,我们又不得不求助电话本,但是,通过电话本查找还是很费时间的.但是,我们大脑能够记住的东西是一定的,我们只能记住自己最熟悉的,而长时间不熟悉的自然就忘记了. 其实,计算机也用到了同样的一个概念,我们用缓存来存放以前读取的数据,而不是直接丢掉,这样,再次读取的时候,可以直接在缓存里面取,而不用再重新查找一遍,这样系统的反应能力

Java 8的五大开发技巧

转载:http://geek.csdn.net/news/detail/94219 在Java 9发布之前,我们来分享一些Java 8开发技巧,本文翻译自JetBrains高级开发主管Trisha Geed的Java 8 Top Tips. [以下为译文] 在使用JAVA 8进行开发多年后,结合个人使用IntelliJ IDEA的心得,我总结了以下几个JAVA8技巧供大家参考. Optional Optional 是一个被高度低估的特性,假若能把它运用恰当,其实它是能帮助减少大量的NullPoi

请用Java设计一个Least Recently Used (LRU) 缓存

LRU介绍:LRU是Least Recently Used的缩写,即最少使用页面置换算法,是为虚拟页式存储管理服务的, 思路介绍: 可以使用两个标准的数据结构来实现,Map和Queue.因为需要支持多线程,需要使用实现了java.utili.concurrent.*的Map和Queue. 主要思路是使用一个Queue来维护FIFO和Map来对数据进行排序,当向缓存添加新的元素时,共有以下三种可能 1. 如果该元素已经在Cache中存在(Map),我们会从queue中删除改元素并将其添加到queu

Java集合详解5:深入理解LinkedHashMap和LRU缓存

Java集合详解5:深入理解LinkedHashMap和LRU缓存 今天我们来深入探索一下LinkedHashMap的底层原理,并且使用linkedhashmap来实现LRU缓存. 具体代码在我的GitHub中可以找到 https://github.com/h2pl/MyTech 文章首发于我的个人博客: https://h2pl.github.io/2018/05/11/collection5 更多关于Java后端学习的内容请到我的CSDN博客上查看:https://blog.csdn.net