百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程字典 > 正文

Android LruCache缓存原理

toyiye 2024-06-21 12:09 13 浏览 0 评论

1.背景

在实际开发中,缓存机制使用最频繁的便是图片缓存!目前大部分的App都是图文结合,从web服务器获取文字和图片,文字显示很快,图片基本上是先下载到手机本地,然后再显示,如果图片很多、很大,每次加载同一张图片,都去网络下载,那么App渲染的速度是比较慢的,这样的体验很差!所以,类似这样的场景,便要使用缓存机制! 目前缓存机制使用大致流程是,当App需要加载某一张图片时,先去手机内存中去找该图片,如果有,那么直接显示,如果无,则去手机sd卡或者手机外部存储中找该图片,如果有,那么直接显示,如果无,那么此时才去网络下载该图片。这种机制常称为三级缓存策略。

2.简介

LruCache这个类在android.util包下,是API level 12引入的,对于API level 12之前的系统可以使用support library中的LruCache。这个类非常适合用来缓存图片,它的主要算法原理是把最近使用的对象用强引用存储在 LinkedHashMap中,并且把最近最少使用的对象在缓存值达到预设定值之前从内存中移除。

3.LruCache的使用

//设置缓存的大小,基本上设置为手机内存的1/8
int maxMemory = (int) Runtime.getRuntime().maxMemory();
int cacheSize = maxMemory / 8;
//LruCache里面的键值对分别是URL和对应的图片
LruCache<String, Bitmap> mCache = new LruCache<String, Bitmap>(cacheSize) {
 @Override
 protected int sizeOf(String key, Bitmap value) {
 //在每次存入缓存的时候调用,计算出要缓存的每张图片的大小
 return value.getByteCount();
 }
};

4.LruCache实现原理

LruCache内部的缓存实际是由LinkedHashMap来维护的,下面是LinkedHashMap的构造函数

public LinkedHashMap(int initialCapacity, //初始容量
 float loadFactor, //加载因子
 boolean accessOrder) { //排序方式,true则按访问顺序,为false,则按插入顺序
 super(initialCapacity, loadFactor);
 this.accessOrder = accessOrder;
}

下面是Lrucache的构造方法

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); //LruCache默认为按访问排序
}

LruCache中的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++; //put方法调用次数加1
 size += safeSizeOf(key, value); //safeSizeOf()方法最终会调用创建LruCache时
 //重写的SizeOf()方法并返回该方法的值
 previous = map.put(key, value); //执行该方法时内部会判断该值是否已存在
 if (previous != null) { //该值若存在会赋值给previous,否则为空
 size -= safeSizeOf(key, previous);
 }
 }
 if (previous != null) {
 entryRemoved(false, key, previous, value); //该方法需要在创建LruCache时重写,若不重写,则不作任何操作。
 }
 trimToSize(maxSize); ////调整缓存大小(关键方法)
 return previous;
}

可以看到put()方法并没有什么难点,重要的就是在添加过缓存对象后,调用 trimToSize()方法,来判断缓存是否已满,如果满了就要删除近期最少使用的算法。

trimToSize()方法

public void trimToSize(int maxSize) {
 while (true) {
 K key;
 V value;
 //如果map为空并且缓存size不等于0或者缓存size小于0,抛出异常
 synchronized (this) {
 if (size < 0 || (map.isEmpty() && size != 0)) {
 throw new IllegalStateException(getClass().getName()
 + ".sizeOf() is reporting inconsistent results!");
 }
 //如果缓存大小size小于最大缓存,或者map为空,不需要再删除缓存对象,跳出循环
 if (size <= maxSize) {
 break;
 }
 //迭代器获取第一个对象,即队尾的元素,近期最少访问的元素
 Map.Entry<K, V> toEvict = map.eldest();
 if (toEvict == null) {
 break;
 }
 //删除该对象,并更新缓存大小
 key = toEvict.getKey();
 value = toEvict.getValue();
 map.remove(key);
 size -= safeSizeOf(key, value);
 evictionCount++;
 }
 entryRemoved(true, key, value, null);
 }
}

LruCache中的get()方法

public final V get(K key) {
 if (key == null) {
 throw new NullPointerException("key == null");
 }
 V mapValue;
 synchronized (this) {
 mapValue = map.get(key); 
 if (mapValue != null) {
 hitCount++; //get获取缓存命中时次数加1
 return mapValue; //获取到值跳出方法返回取到的值
 }
 missCount++; //get获取缓存未命中时次数加1
 }
 V createdValue = create(key); //create()方法需要重写,不重写则返回null
 if (createdValue == null) {
 return null;
 }
 synchronized (this) {
 createCount++; //create()方法成功调用次数加1
 mapValue = map.put(key, createdValue);
 if (mapValue != null) {
 // There was a conflict so undo that last put
 map.put(key, mapValue);
 } else {
 size += safeSizeOf(key, createdValue);
 }
 }
 if (mapValue != null) {
 entryRemoved(false, key, createdValue, mapValue);
 return mapValue;
 } else {
 trimToSize(maxSize);
 return createdValue;
 }
}

LinkedHashMap中的get()方法

public V get(Object key) {
 LinkedHashMapEntry<K,V> e = (LinkedHashMapEntry<K,V>)getEntry(key);
 if (e == null)
 return null;
 e.recordAccess(this); //实现排序的关键方法
 return e.value;
}

$emsp;recordAccess()方法

void recordAccess(HashMap<K,V> m) {
 LinkedHashMap<K,V> lm = (LinkedHashMap<K,V>)m;
 //判断是否是访问排序
 if (lm.accessOrder) {
 lm.modCount++;
 remove(); //删除此元素
 addBefore(lm.header); //将此元素移动到队列的头部
 }
}

以上便是LruCache实现的原理,理解了LinkedHashMap的数据结构就能理解整个原理。如果不懂,可以先看看LinkedHashMap的具体实现。

相关推荐

为何越来越多的编程语言使用JSON(为什么编程)

JSON是JavascriptObjectNotation的缩写,意思是Javascript对象表示法,是一种易于人类阅读和对编程友好的文本数据传递方法,是JavaScript语言规范定义的一个子...

何时在数据库中使用 JSON(数据库用json格式存储)

在本文中,您将了解何时应考虑将JSON数据类型添加到表中以及何时应避免使用它们。每天?分享?最新?软件?开发?,Devops,敏捷?,测试?以及?项目?管理?最新?,最热门?的?文章?,每天?花?...

MySQL 从零开始:05 数据类型(mysql数据类型有哪些,并举例)

前面的讲解中已经接触到了表的创建,表的创建是对字段的声明,比如:上述语句声明了字段的名称、类型、所占空间、默认值和是否可以为空等信息。其中的int、varchar、char和decimal都...

JSON对象花样进阶(json格式对象)

一、引言在现代Web开发中,JSON(JavaScriptObjectNotation)已经成为数据交换的标准格式。无论是从前端向后端发送数据,还是从后端接收数据,JSON都是不可或缺的一部分。...

深入理解 JSON 和 Form-data(json和formdata提交区别)

在讨论现代网络开发与API设计的语境下,理解客户端和服务器间如何有效且可靠地交换数据变得尤为关键。这里,特别值得关注的是两种主流数据格式:...

JSON 语法(json 语法 priority)

JSON语法是JavaScript语法的子集。JSON语法规则JSON语法是JavaScript对象表示法语法的子集。数据在名称/值对中数据由逗号分隔花括号保存对象方括号保存数组JS...

JSON语法详解(json的语法规则)

JSON语法规则JSON语法是JavaScript对象表示法语法的子集。数据在名称/值对中数据由逗号分隔大括号保存对象中括号保存数组注意:json的key是字符串,且必须是双引号,不能是单引号...

MySQL JSON数据类型操作(mysql的json)

概述mysql自5.7.8版本开始,就支持了json结构的数据存储和查询,这表明了mysql也在不断的学习和增加nosql数据库的有点。但mysql毕竟是关系型数据库,在处理json这种非结构化的数据...

JSON的数据模式(json数据格式示例)

像XML模式一样,JSON数据格式也有Schema,这是一个基于JSON格式的规范。JSON模式也以JSON格式编写。它用于验证JSON数据。JSON模式示例以下代码显示了基本的JSON模式。{"...

前端学习——JSON格式详解(后端json格式)

JSON(JavaScriptObjectNotation)是一种轻量级的数据交换格式。易于人阅读和编写。同时也易于机器解析和生成。它基于JavaScriptProgrammingLa...

什么是 JSON:详解 JSON 及其优势(什么叫json)

现在程序员还有谁不知道JSON吗?无论对于前端还是后端,JSON都是一种常见的数据格式。那么JSON到底是什么呢?JSON的定义...

PostgreSQL JSON 类型:处理结构化数据

PostgreSQL提供JSON类型,以存储结构化数据。JSON是一种开放的数据格式,可用于存储各种类型的值。什么是JSON类型?JSON类型表示JSON(JavaScriptO...

JavaScript:JSON、三种包装类(javascript 包)

JOSN:我们希望可以将一个对象在不同的语言中进行传递,以达到通信的目的,最佳方式就是将一个对象转换为字符串的形式JSON(JavaScriptObjectNotation)-JS的对象表示法...

Python数据分析 只要1分钟 教你玩转JSON 全程干货

Json简介:Json,全名JavaScriptObjectNotation,JSON(JavaScriptObjectNotation(记号、标记))是一种轻量级的数据交换格式。它基于J...

比较一下JSON与XML两种数据格式?(json和xml哪个好)

JSON(JavaScriptObjectNotation)和XML(eXtensibleMarkupLanguage)是在日常开发中比较常用的两种数据格式,它们主要的作用就是用来进行数据的传...

取消回复欢迎 发表评论:

请填写验证码