热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

AndroidBitmap缓存策略

Android中最常用到缓存的地方就是图片,通过过缓存即可以提高应用程序的效率,又可以节省用户的流量。图片的缓存简单来说可以分为SD卡缓存和内存缓存,也可以俩者配合使用。Andr

Android中最常用到缓存的地方就是图片,通过过缓存即可以提高应用程序的效率,又可以节省用户的流量。

图片的缓存简单来说可以分为SD卡缓存和内存缓存,也可以俩者配合使用。
Android中图片缓存遵循的策略就是:当第一次从网络中加载图片的时候,将其缓存到存储设备上(比如sd卡,这也就是我们说的SD卡缓存),并且在内存中同样也缓存一份(内存缓存),这样当下次使用或者网络请求图片的时候,就先去内存中获取(因为从内存中读取要比从SD卡中读取的快,所以我们先从内存中读取,这样的话在节省流量的同时也可以提高程序的性能),如果内存中没有的话再去SD卡中获取,如果SD卡中也没有的话才去网络上请求。

当然,由于设备的存储容量限制,我们也需要把缓存到SD卡的图片资源在一个合适的时间内删除。当然,这个合适的时间怎么定义就仁者见仁智者见智了,目前常用的是算法是LRU算法,所以一个完整的缓存策略包括添加,获取,删除三类操作

LRU缓存算法

LRU(Least Recently Used),近期最少使用算法,它的核心思想就是当缓存满时,优先淘汰即删除近期最少使用的缓存对象。
采用LRU算法的缓存有俩种:

  • LruCache,用于实现内存缓存
  • DiskLruCache,用于实现存储设备缓存(它不属于官方SDK的一部分,但得到官方的推荐)
  • 其他存储方式,比如文件存储。SqlLite存储等

LruCache

LruCache是Android 3.1 提供的,使用support-4兼容包中LruCache可以向下兼容到2.2。
LruCache内部采用一个LinkedHashMap以强引用的方式存储外界提供的缓存对象,提供了put和get方法来添加和获取缓存对象,另外LruCache也是线程安全的。。

1、LruCache的使用

Runtime.getRuntime().maxMemory()在Java中返回的是java虚拟机(这个进程)能构从操纵系统那里挖到的最大的内存,而在Android中返回应用程序最大可用内存,以字节为单位。也可以用
((ActivityManager)getSystemService(Context.ACTIVITY_SERVICE)).getMemoryClass()来获取,不过返回值的单位是M。
Runtime.getRuntime().totalMemory()在Java中返回的是java虚拟机现在已经从操纵系统那里挖过来的内存大小,也就是java虚拟机这个进程当时所占用的所有内存,在Android中返回的是应用程序已获得内存,所以totalMemory()是慢慢增大的。
Runtime.getRuntime().freeMemory()简单来说就是已经获取到但是还没有使用的内存。
创建LruCache代码:

//创建LruCache对象
int maxSize = (int) Runtime.getRuntime().maxMemory();// 返回byte
int cacheSize = maxSize / 1024 / 8;
// 创建LruCache时需要提供缓存的最大容量
mCache = new LruCache(cacheSize) {
    @Override
    protected int sizeOf(String key, Bitmap value) {
        return super.sizeOf(key, value);
    }
    @Override
    protected void entryRemoved(boolean evicted, String key, Bitmap oldValue, Bitmap newValue) {
        // 当调用put或remove时触发此方法,可以在这里完成一些资源回收的操作
        super.entryRemoved(evicted, key, oldValue, newValue);
    }
};
mCache.put(key, value);// 添加缓存对象
mCache.get(key);// 获取缓存对象
mCache.remove(key);// 删除缓存对象
mCache.resize(1024);// 重新设置缓存的总容量大小

DiskLruCache

上面讲过,DiskLruCache不属于Android SDK 的一部分,但是它得到了Google官方的认可与推荐,它的源码在这里或者这里;其实GitHub上才是正儿八经的出产地哈。在使用DiskLruCache时,首先要从网上下载DiskLruCache的源码,然后放到自己的项目里编译后才能正常使用。

1、DiskLruCache 的创建

DiskLruCache不能通过构造方法来创建,它通过open方法来穿件自身
public static DiskLruCache open(File directory, int appVersion, int valueCount, long maxSize)
1. directory表示磁盘缓存的存储路径
缓存目录没有具体限制,可以根据需求自己的定义。一般来说,可以选择SD卡上的/sdcard/Android/data//cache目录,这个目录是Android系统指定的应用程序缓存目录,当应用卸载时,缓存也会被系统清除;当然还可以选择sd卡上的其他目录,也可以选择data下的当前应用目录。当然,一个严禁的程序还要考虑SD卡是否存在等。
2. appVersion表示应用的版本号
当appVersion改变时,之前的缓存都会被清除,所以如非必要,我们为其指定一个1,不再改变即可
3. valueCount表示单个节点对应的数据个数,也就是同一个key可以对应多少个缓存文件,一般来说我们都选取1.
4. maxSize缓存的总大小。

 private void createDiskLruCache() {
     try {
         File cacheDir = getDiskCacheDir(DiskLruCacheActivity.this, "bitmapsCache");
         if (!cacheDir.exists()) {
             cacheDir.mkdirs();
         }
         diskLruCache = DiskLruCache.open(cacheDir, 1, 1, MAX_DISK_CACHE);
     } catch (IOException e) {
         Log.i("disk cache", "createDiskLruCache e: " + e.toString());
     }
 }
2、DiskLruCache 添加缓存

DiskLruCache的缓存添加是通过Editor完成的,代码如下:

 // 把图片添加到硬盘缓存
 private void exeAdd2DiskCache(DiskLruCache.Editor editor) {
     if (editor != null) {
         try {
             // 创建DiskLruCache时设置一个节点只有一个数据,所以这里的index直接设为0即可
             OutputStream outputStream = editor.newOutputStream(0);
             if (downloadUrlToStream(imageUrl, outputStream)) {
                 editor.commit();// 提交写入操作
                 Log.i("disk cache", "onCreate editor.commit() ");
             } else {
                 editor.abort();// 回退整个操作
                 Log.i("disk cache", "onCreate editor.abort() ");
             }
             diskLruCache.flush();
         } catch (IOException e) {
             Log.i("disk cache", "onCreate e: " + e.toString());
         }
     }
 }

 // 建立HTTP请求,并获取Bitmap对象。
 private boolean downloadUrlToStream(String urlString, OutputStream outputStream) {
     HttpURLConnection urlCOnnection= null;
     BufferedOutputStream out = null;
     BufferedInputStream in = null;
     try {
         final URL url = new URL(urlString);
         urlCOnnection= (HttpURLConnection) url.openConnection();
         in = new BufferedInputStream(urlConnection.getInputStream(), 8 * 1024);
         out = new BufferedOutputStream(outputStream, 8 * 1024);
         int b;
         while ((b = in.read()) != -1) {
             out.write(b);
         }
         return true;
     } catch (final IOException e) {
         e.printStackTrace();
     } finally {
         if (urlConnection != null) {
             urlConnection.disconnect();
         }
         try {
             if (out != null) {
                 out.close();
             }
             if (in != null) {
                 in.close();
             }
         } catch (final IOException e) {
             e.printStackTrace();
         }
     }
     return false;
 }

当然,以上只是展示DiskLruCache的用法,在实际项目中还要和项目需求结合,考虑实际情况,比如Caused by: android.os.NetworkOnMainThreadException异常等。

3、DiskLruCache 从缓存中查找

首先需要将URL转换成Key,然后通过DiskLruCache的get方法得到一个Snapshot对象,在通过Snapshot对象得到缓存文件的输入流,再把输入流转换成Bitamp对象。

 // 从缓存中获取Bitmap对象
 private Bitmap getCacheBitmap() {
     String key = hashKeyForDisk(imageUrl);// 把Url转换成KEY
     try {
         DiskLruCache.Snapshot snapShot = diskLruCache.get(key);// 通过key获取Snapshot对象
         if (snapShot != null) {
             InputStream is = snapShot.getInputStream(0);// 通过Snapshot对象获取缓存文件的输入流
             Bitmap bitmap = BitmapFactory.decodeStream(is);// 把输入流转换成Bitmap对象
             return bitmap;
         }
     } catch (IOException e) {
         e.printStackTrace();
     }
     return null;
 }

关于 BitmapFactory.decodeStream(is) 更多资料请点这里查看

4、DiskLruCache 缓存的删除
    private void deleteCacheBitmap(String url){
        try {
            String key = hashKeyForDisk(imageUrl);
            diskLruCache.remove(key);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

这个remove方法,如果不是必要的话,我们尽量不要去调用,因为我们已经设定了最大缓存容量,当超过这个容量时,系统会根据Lru算法自动删除该删除的缓存文件。

5、DiskLruCache 其他几个重要且常用的方法
  1. diskLruCache.delete() 删除所有的缓存数据
  2. diskLruCache.delete() 返回当前缓存路径下所有缓存数据的大小,以byte为单位
  3. editor.abort() 图片下载发生异常时,可以通过这个方法回退整个操作
  4. editor.commit() 提交写入操作,这个放在在写入缓存数据时是一定要调用的
  5. diskLruCache.flush() 这个方法用于将内存中的操作记录同步到日志文件(也就是journal文件,系统LRU算法依赖于这个文件,因为这个文件中保存着对数据的操作记录)当中;但其实并不是每次写入缓存都要调用一次flush()方法的,频繁地调用并不会带来任何好处,只会额外增加同步journal文件的时间。比较标准的做法就是在Activity的onPause()方法中去调用一次flush()方法就可以了。
  6. diskLruCache.close() 这个方法用于将DiskLruCache关闭掉,是和open()方法对应的一个方法。关闭掉了之后就不能再调用DiskLruCache中任何操作缓存数据的方法,通常只应该在Activity的onDestroy()方法中去调用close()方法。
6、journal 文件

当执行完写入操作后,我们看看对应的目录下(/sdcard/Android/data//cache)有什么文件?

打开cache目录后,发现里边只有一个bitmapsCache的文件夹;这个文件是哪里来的呢?这个其实就是上文在创建DiskLruCache实例时传入的url中拼接的(看 getDiskCacheDir 方法),为什么要指定这么一个目录呢?其实就是类似于分类的概念,比如你可以把缓存的Bitmap放到一个文件夹下,把file或者其他格式的数据放到另外一个文件夹下。

打开bitmapsCache文件夹,它的子目录又有哪些呢?首先有一个文件,这个文件的文件名很长而且没有任何规则,完全看不懂是什么意思;另外下边还有一个journal文件。其实文件名很长的文件就是一张缓存的图片,每个文件都对应着一张图片,如果我们缓存了很多图片的话,就会有一堆这样的文件;而journal文件是DiskLruCache的一个日志文件,就像我上面说的:这个文件中保存着对数据的操作记录。如下图:

打开journal这个文件,发现它长成这样 长得也很整齐:
首先有一行字符串“libcore.io.DiskLruCache”表示我们使用的是DiskLruCache技术;然后又有三行,且每行都只有一个“1”,其中第一行的“1”表示DiskLruCache的版本号,这个值是恒为1的,第2行的“1”表示应用程序的版本号,我们在open()方法里传入的版本号是什么这里就会显示什么,第三个“1”表示的是valueCount,表示单个节点对应的数据个数,这个值也是在open()方法中传入的,通常情况下都为1。接下来就是一个空行,标志着开始记录数据的操作记录。

接下来,会有DIRTY开头的一行数据,DIRTY后边跟着的是文件的key,DIRTY表示开始向缓存中写入数据,但写入结果是什么还未知。然后调用commit()方法表示写入缓存成功,这时会向journal中写入一条CLEAN记录,表示数据写入成功;如果数据写入失败,会调用abort()方法回退整个操作,这时会向journal中写入一条REMOVE记录。当调用get()方法去读取一条缓存数据时,就会向journal文件中写入一条READ记录;另外,某些行后面还有一个数字(20090、6602),这个数字就是缓存图片的大小,以byte为单位。

===============================================
上一篇:Bitmap的使用
更多内容请关注 我的专题
转载请注明 原文链接:
http://www.jianshu.com/p/635fceca82d3

===============================================
这天,在简书上看到一篇帖子,里边有几张图片觉着挺好看,发出来,一起瞅瞅 :)








推荐阅读
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • 本文介绍了Java工具类库Hutool,该工具包封装了对文件、流、加密解密、转码、正则、线程、XML等JDK方法的封装,并提供了各种Util工具类。同时,还介绍了Hutool的组件,包括动态代理、布隆过滤、缓存、定时任务等功能。该工具包可以简化Java代码,提高开发效率。 ... [详细]
  • Java String与StringBuffer的区别及其应用场景
    本文主要介绍了Java中String和StringBuffer的区别,String是不可变的,而StringBuffer是可变的。StringBuffer在进行字符串处理时不生成新的对象,内存使用上要优于String类。因此,在需要频繁对字符串进行修改的情况下,使用StringBuffer更加适合。同时,文章还介绍了String和StringBuffer的应用场景。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • 本文介绍了一个在线急等问题解决方法,即如何统计数据库中某个字段下的所有数据,并将结果显示在文本框里。作者提到了自己是一个菜鸟,希望能够得到帮助。作者使用的是ACCESS数据库,并且给出了一个例子,希望得到的结果是560。作者还提到自己已经尝试了使用"select sum(字段2) from 表名"的语句,得到的结果是650,但不知道如何得到560。希望能够得到解决方案。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • 本文介绍了南邮ctf-web的writeup,包括签到题和md5 collision。在CTF比赛和渗透测试中,可以通过查看源代码、代码注释、页面隐藏元素、超链接和HTTP响应头部来寻找flag或提示信息。利用PHP弱类型,可以发现md5('QNKCDZO')='0e830400451993494058024219903391'和md5('240610708')='0e462097431906509019562988736854'。 ... [详细]
  • 本文介绍了如何使用C#制作Java+Mysql+Tomcat环境安装程序,实现一键式安装。通过将JDK、Mysql、Tomcat三者制作成一个安装包,解决了客户在安装软件时的复杂配置和繁琐问题,便于管理软件版本和系统集成。具体步骤包括配置JDK环境变量和安装Mysql服务,其中使用了MySQL Server 5.5社区版和my.ini文件。安装方法为通过命令行将目录转到mysql的bin目录下,执行mysqld --install MySQL5命令。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • 拥抱Android Design Support Library新变化(导航视图、悬浮ActionBar)
    转载请注明明桑AndroidAndroid5.0Loollipop作为Android最重要的版本之一,为我们带来了全新的界面风格和设计语言。看起来很受欢迎࿰ ... [详细]
  • 利用Visual Basic开发SAP接口程序初探的方法与原理
    本文介绍了利用Visual Basic开发SAP接口程序的方法与原理,以及SAP R/3系统的特点和二次开发平台ABAP的使用。通过程序接口自动读取SAP R/3的数据表或视图,在外部进行处理和利用水晶报表等工具生成符合中国人习惯的报表样式。具体介绍了RFC调用的原理和模型,并强调本文主要不讨论SAP R/3函数的开发,而是针对使用SAP的公司的非ABAP开发人员提供了初步的接口程序开发指导。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • 在Java中,我会做这样的事情:classPerson{privateRecordrecord;publicStringname(){record().get(name);}p ... [详细]
  • 本文介绍了将mysql从5.6.15升级到5.7.15的详细步骤,包括关闭访问、备份旧库、备份权限、配置文件备份、关闭旧数据库、安装二进制、替换配置文件以及启动新数据库等操作。 ... [详细]
author-avatar
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有