LruCache缓存图片研究小结

2024-05-14 11:38

本文主要是介绍LruCache缓存图片研究小结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一篇研究了LinkedHashMap实现LRU策略,虽然通过上述方式来实现图片缓存可以优化内存的使用效率,但是这种方式也存在一些问题,例如,LinkedHashMap不是线程安全的,所以在操作时需要考虑线程安全问题。另外在缓存时,只能指定缓存数据条目的数量,不能指定缓存区的大小,如果需要缓存的图片都比较大,可能就会出现问题。。。。

其实在Android SDK 中已经为我们提供了一个实现LRU策略的cache类,LruCache类,这个类封装了LinkedHashMap并且解决了LinkedHashMap中存在的问题,今天我们以LruCache缓存图片为例研究下这个类。

有一点需要提前说明下,今天我们研究的是android.support.v4.util包中的LruCache类而不是android.util包中也有一个LruCache类,这点需要提前说明下,不然很容易蒙圈,因为这两个类中有些方法实现是不同的。

首先我们看一下LruCache的成员变量,代码如下

    private final LinkedHashMap<K, V> map;private int size;//当前容量private int maxSize;//最大容量private int putCount;//put的次数 private int createCount;//create的次数 private int evictionCount;//回收次数private int hitCount;//命中次数private int missCount;//丢失的次数

再看下构造方法。

   public LruCache(int maxSize) {if (maxSize <= 0) {throw new IllegalArgumentException("maxSize <= 0");}this.maxSize = maxSize;this.map = new LinkedHashMap<K, V>(0, 0.75f, true);}

在构造方法中初始化了maxSize和LinkedHashMap,并且LinkedHashMap为按照访问顺序排序(此部分请参照上一篇文章)。maxSize即为最大容量。

前面提到LruCache可以指定缓存区大小,这怎么实现呢?。。其实实现起来很简单只要在初始化的时候重写LruCache提供的sizeOf方法即可,代码如下。

        int LRU_CACHE_SIZE = 4 * 1024 * 1024; //4MBLruCache<String,Bitmap> lruCache=new LruCache<String,Bitmap>(LRU_CACHE_SIZE) {@Overrideprotected int sizeOf(String key, Bitmap value) {if (value != null)return value.getByteCount();elsereturn 0;}};

首先我们定义了LruCache的最大容量为4 * 1024 * 1024,即4M的空间,然后我们重写sizeOf方法,让这个方法返回Bitmap的字节数。为什么这样写就可以指定缓存区大大小呢?接下来我们看下比较重要的put和get方法,你就会明白了~

首先看下put方法的源码。

   public final V put(K key, V value) {if (key == null || value == null) {throw new NullPointerException("key == null || value == null");}V previous;synchronized (this) {putCount++;size += safeSizeOf(key, value);previous = map.put(key, value);if (previous != null) {size -= safeSizeOf(key, previous);}}if (previous != null) {entryRemoved(false, key, previous, value);}trimToSize(maxSize);return previous;}

在方法体中,我们发现当key或者value为空时会抛出异常,这部分和LinkedHashMap是不同的,说明LruCache类是不支持key或者value为空,并且在对map进行put操作时加了synchronized ,保证了线程安全。另外我们发现有如下操作

size += safeSizeOf(key, value);

这部分看上去像对size(当前容量)的累加,我们继续进入到safeSizeOf方法中

   private int safeSizeOf(K key, V value) {int result = sizeOf(key, value);if (result < 0) {throw new IllegalStateException("Negative size: " + key + "=" + value);}return result;}

在这个方法中看到了之前重写sizeOf方法,看到这里我想大家就明白了。我们之前重写sizeOf方法让它返回bitmap的字节数,而在put方法中会对每次put进来的bimap的字节数利用size进行累加,这样我们利用size和maxSize进行比较,就可以得知当前容量是否大于最大容量,当大于时就可以进行删除最近最少少用的资源的操作了。另外sizeOf默认情况下是返回1的,所以如果不重写sizeOf方法,LruCache也是进行“计数”的。

接着看put方法的代码,我们会发现如下操作。

       previous = map.put(key, value);if (previous != null) {size -= safeSizeOf(key, previous);}

我们都知道,在对一个hashmap进行put操作的时候 ,如果put的key-value键值对中的key已经存在与map中,新put的value会覆盖旧value,并且会返回旧value。如果key不存在于map中,则会返回null。这里利用previous获取返回值,如果previous不为空,则当前容量会减去previous的大小,这部分比较好理解,其实就是防止同一个key对应新旧value的大小的重复叠加。

接着往下看,在previous不为空的时候,会调用 entryRemoved(false, key, previous, value) 方法,并且把previous传进去了,查看此方法,我们发现此方法并没有函数体,看来需要我们重写此方法。

protected void entryRemoved(boolean evicted, K key, V oldValue, V newValue) {}

根据前面所讲,我们可以在初始化的时候重写这个方法,来释放掉一个key对应的旧value所占用的资源,代码如下。

 LruCache<String,Bitmap> lruCache=new LruCache<String,Bitmap>(LRU_CACHE_SIZE) {@Overrideprotected int sizeOf(String key, Bitmap value) {if (value != null)return value.getByteCount();elsereturn 0;}@Overrideprotected void entryRemoved(boolean evicted, String key, Bitmap oldValue, Bitmap newValue) {if(evicted==false&&oldValue!=null){oldValue.recycle();//释放bitmap资源}}};

看到这里,可能会有疑问,在什么方法里会进行删除最近最少的操作呢?接着看put方法我们会发现trimToSize方法,答案其实就在这个方法里面。代码如下。

public void trimToSize(int maxSize) {while (true) {//不断循环删除linkedHashMap首元素,也就是最近最少访问的条目,直到size小于最大容量或者map中已经没有数据K key;V value;synchronized (this) {if (size < 0 || (map.isEmpty() && size != 0)) {throw new IllegalStateException(getClass().getName()+ ".sizeOf() is reporting inconsistent results!");}if (size <= maxSize || map.isEmpty()) {//直到当前容量小于最大容量 break;}Map.Entry<K, V> toEvict = map.entrySet().iterator().next();//指向首元素key = toEvict.getKey();value = toEvict.getValue();map.remove(key);//删除最近最少的entrysize -= safeSizeOf(key, value);evictionCount++;}entryRemoved(true, key, value, null);}}

观察方法体我们发现,当当前容量大于最大容量时,会不断删除首元素即最近最少访问的元素,然后重新计算当前容量大小。只有在当前容量小于最大容量或者map中没有数据的时候才会break出去。在代码的最后也会调用一次entryRemoved(true, key, value, null)方法,并且把因为空间不足而删除的元素的key和value传递进去,不过第一参数为true有别与先前调用时传递的false,所以根据第一参数区别我们就可以进行一些不一样的操作。

protected void entryRemoved(boolean evicted, String key, Bitmap oldValue, Bitmap newValue) {if(evicted==false&&oldValue!=null){oldValue.recycle();}if(evicted==true&&oldValue!=null){//TODO 根据key value 进行二级缓存}}

好了,put方法说明的比较详细,相信大家已经有了比较深的了解,接下来get方法我们加快点节奏。get方法源码如下。

public final V get(K key) {if (key == null) {//key 不允许为空throw new NullPointerException("key == null");}V mapValue;synchronized (this) {mapValue = map.get(key);if (mapValue != null) {hitCount++;return mapValue;}missCount++;}V createdValue = create(key);//根据key进行新建if (createdValue == null) {return null;}synchronized (this) {createCount++;mapValue = map.put(key, createdValue);if (mapValue != null) {//有返回值,说明key对应value已存在,需要进行重现赋值,即取消上一步操作map.put(key, mapValue);} else {size += safeSizeOf(key, createdValue);}}if (mapValue != null) {entryRemoved(false, key, createdValue, mapValue);//可以释放刚创建的createdValuereturn mapValue;} else {trimToSize(maxSize);return createdValue;}}

观察代码,我们发现其实lrucache方法中主要是对linkedhashmap进行get操作,这部分不清除的同学可以看下上一篇关于linkedhashmap的研究总结。

我们重点研究下通过key获取到的value为空时的case。观察代码,如果如果出现上述情况,会新建一个createdValue,并且把createdValue put到map中,利用mapValue 获取put返回值,继续判断mapValue 是否为空,如果mapValue 不为空说明此时此key在map中存在对应的value,所以接下来需要进行重新赋值,让key对应的value为mapValue 而不是createdValue。如果mapValue为空,说明key对应的value确实为空,需要进行就是重新计算当前size的大小。在代码的最后会判断mapValue是否为空,如果mapValue不为空的时候,会调用entryRemoved方法,并把createdValue放在oldvalue的位置,因为此时createdValue所占的资源是无用的,所以我们可以在entryRemoved中释放createdValue。如果mapValue不为空又会进入到trimToSize方法中进行容量计算和删除“最近最少”。好了,get也方法讲完了,最后看下remove方法。

   public final V remove(K key) {if (key == null) {throw new NullPointerException("key == null");}V previous;synchronized (this) {previous = map.remove(key);if (previous != null) {size -= safeSizeOf(key, previous);}}if (previous != null) {entryRemoved(false, key, previous, null);}return previous;}

观察代码,基本上和get和put有异曲同工之妙,在这里就不重复阐述了,大家自己阅读代码吧!~~~

比较重要的方法都讲完了,到此大家对lrucache应该有了一个全面的了解了吧~~接下来我们稍微总结下lrucache吧。

1.LruCache封装了LinkedHashMap,提供了LRU缓存的功能,并且在关键操作加了synchronized ,实现了线程安全。
2.LruCache提供了trimToSize方法,当容量不足时会自动删除最近最少访问的键值对。
3.LruCache提供了entryRemoved(boolean evicted, K key, V oldValue, V newValue)方法,通过重写这个方,结合put,get,remove方法我们可以做更多的事情。
3.LruCache不允许空键值;
5.LruCache提供了sizeof方法,重写这个方法可以实现指定缓存区大小,而不是像LinkedHashMap一样只能指定缓存条目数。

ok!就写到这了,欢迎大家一起研究交流!~~~

这篇关于LruCache缓存图片研究小结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/988648

相关文章

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

缓存雪崩问题

缓存雪崩是缓存中大量key失效后当高并发到来时导致大量请求到数据库,瞬间耗尽数据库资源,导致数据库无法使用。 解决方案: 1、使用锁进行控制 2、对同一类型信息的key设置不同的过期时间 3、缓存预热 1. 什么是缓存雪崩 缓存雪崩是指在短时间内,大量缓存数据同时失效,导致所有请求直接涌向数据库,瞬间增加数据库的负载压力,可能导致数据库性能下降甚至崩溃。这种情况往往发生在缓存中大量 k

Android 10.0 mtk平板camera2横屏预览旋转90度横屏拍照图片旋转90度功能实现

1.前言 在10.0的系统rom定制化开发中,在进行一些平板等默认横屏的设备开发的过程中,需要在进入camera2的 时候,默认预览图像也是需要横屏显示的,在上一篇已经实现了横屏预览功能,然后发现横屏预览后,拍照保存的图片 依然是竖屏的,所以说同样需要将图片也保存为横屏图标了,所以就需要看下mtk的camera2的相关横屏保存图片功能, 如何实现实现横屏保存图片功能 如图所示: 2.mtk

Spring MVC 图片上传

引入需要的包 <dependency><groupId>commons-logging</groupId><artifactId>commons-logging</artifactId><version>1.1</version></dependency><dependency><groupId>commons-io</groupId><artifactId>commons-

Prompt - 将图片的表格转换成Markdown

Prompt - 将图片的表格转换成Markdown 0. 引言1. 提示词2. 原始版本 0. 引言 最近尝试将图片中的表格转换成Markdown格式,需要不断条件和优化提示词。记录一下调整好的提示词,以后在继续优化迭代。 1. 提示词 英文版本: You are an AI assistant tasked with extracting the content of

分布式系统的个人理解小结

分布式系统:分的微小服务,以小而独立的业务为单位,形成子系统。 然后分布式系统中需要有统一的调用,形成大的聚合服务。 同时,微服务群,需要有交流(通讯,注册中心,同步,异步),有管理(监控,调度)。 对外服务,需要有控制的对外开发,安全网关。

Redis中使用布隆过滤器解决缓存穿透问题

一、缓存穿透(失效)问题 缓存穿透是指查询一个一定不存在的数据,由于缓存中没有命中,会去数据库中查询,而数据库中也没有该数据,并且每次查询都不会命中缓存,从而每次请求都直接打到了数据库上,这会给数据库带来巨大压力。 二、布隆过滤器原理 布隆过滤器(Bloom Filter)是一种空间效率很高的随机数据结构,它利用多个不同的哈希函数将一个元素映射到一个位数组中的多个位置,并将这些位置的值置

一种改进的red5集群方案的应用、基于Red5服务器集群负载均衡调度算法研究

转自: 一种改进的red5集群方案的应用: http://wenku.baidu.com/link?url=jYQ1wNwHVBqJ-5XCYq0PRligp6Y5q6BYXyISUsF56My8DP8dc9CZ4pZvpPz1abxJn8fojMrL0IyfmMHStpvkotqC1RWlRMGnzVL1X4IPOa_  基于Red5服务器集群负载均衡调度算法研究 http://ww

生信圆桌x生信分析平台:助力生物信息学研究的综合工具

介绍 少走弯路,高效分析;了解生信云,访问 【生信圆桌x生信专用云服务器】 : www.tebteb.cc 生物信息学的迅速发展催生了众多生信分析平台,这些平台通过集成各种生物信息学工具和算法,极大地简化了数据处理和分析流程,使研究人员能够更高效地从海量生物数据中提取有价值的信息。这些平台通常具备友好的用户界面和强大的计算能力,支持不同类型的生物数据分析,如基因组、转录组、蛋白质组等。

开题报告中的研究方法设计:AI能帮你做什么?

AIPaperGPT,论文写作神器~ https://www.aipapergpt.com/ 大家都准备开题报告了吗?研究方法部分是不是已经让你头疼到抓狂? 别急,这可是大多数人都会遇到的难题!尤其是研究方法设计这一块,选定性还是定量,怎么搞才能符合老师的要求? 每次到这儿,头脑一片空白。 好消息是,现在AI工具火得一塌糊涂,比如ChatGPT,居然能帮你在研究方法这块儿上出点主意。是不