DiskLruCache和LruCache不同的是,LruCache是内存缓存,而DiskLruCache是指磁盘缓存,顾名思义就是把文件缓存到磁盘,也也就是手机的内存卡中。接下来先简单介绍DiskLruCache的使用方法。

下载源码

然后把源码中的三个类拷贝到工程中。

DiskLruCache常用方法:

方法

备注

DiskLruCache open(File directory, int appVersion, int valueCount, long maxSize)

打开一个缓存目录,如果没有则首先创建它,directory:指定数据缓存地址 appVersion:APP版本号,当版本号改变时,缓存数据会被清除 valueCount:同一个key可以对应多少文件 maxSize:最大可以缓存的数据量

Editor edit(String key)

通过key可以获得一个DiskLruCache.Editor,通过Editor可以得到一个输出流,进而缓存到本地存储上

void flush()

强制缓冲文件保存到文件系统

Snapshot get(String key)

通过key值来获得一个Snapshot,如果Snapshot存在,则移动到LRU队列的头部来,通过Snapshot可以得到一个输入流InputStream

long size()

缓存数据的大小,单位是byte

boolean remove(String key)

根据key值来删除对应的数据,如果该数据正在被编辑,则不能删除

void delete()

关闭缓存并且删除目录下所有的缓存数据,即使有的数据不是由DiskLruCache 缓存到本目录的

void close()

关闭DiskLruCache,缓存数据会保留在外存中

boolean isClosed()

判断DiskLruCache是否关闭,返回true表示已关闭

File getDirectory()

缓存数据的目录

初始化缓存对象

接下来具体介绍DiskLruCache的简单方法。首先我们在使用某个类的时候,一般都是首先找到它的构造方法,但是我们发现该类是final 类,无法被继承,并且构造方法是私有的方法,不能手动调用。

public final class DiskLruCache implements Closeable {

private DiskLruCache(File directory, int appVersion, int valueCount, long maxSize) {

this.directory = directory;

this.appVersion = appVersion;

this.journalFile = new File(directory, JOURNAL_FILE);

this.journalFileTmp = new File(directory, JOURNAL_FILE_TEMP);

this.journalFileBackup = new File(directory, JOURNAL_FILE_BACKUP);

this.valueCount = valueCount;

this.maxSize = maxSize;

}

所以在初始化DiskLruCache的时候调用它的open方法

//四个参数分别为,1.缓存的路径目录 2.版本号 3.每个节点对应的数据个数,4.缓存的大小,10 * 1024 * 1024 = 10M

DiskLruCache diskLruCache = DiskLruCache.open(getCachFile(context, uniqueName), 1, 1, cacheSize);

/**

* 获取缓存目录

*

* @param context

* @param uniqueName 指定目录下的文件名

*/

private File getCachFile(Context context, String uniqueName) {

String catchPath;

//有内存卡,并且内存卡没有正在移除,就把文件缓存到内存卡中

if (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState()) || !Environment.isExternalStorageRemovable()) {

catchPath = context.getExternalCacheDir().getPath();

} else {

catchPath = context.getCacheDir().getPath();

}

return new File(catchPath + File.separator + uniqueName);

}

要传入四个参数:

File directory:sdcard缓存的目录。

int appVersion:版本号,一般传1即可

int valueCount:缓存的数据由key对应着,表示一个key对应多少个数据,一般传1即可

long maxSiz:缓存的大小 10 * 1024 * 1024 = 10M

传入sdcard缓存的目录的时候,记得先判断sdcard是否存在,或者sdcard是否正在移除。如果是这两种情况。缓存目录就设置为getCacheDir().getPath();在内存中缓存。

写入缓存

初始化缓存完成之后,就写入缓存,这个时候需要从网上下载一张图片。

new Thread() {

@Override

public void run() {

DiskLruCache.Editor editor = null;

try {

//创建 Editor 对象

editor = diskLruCache.edit(hashKeyForDisk(url));

if (editor != null) {

//创建输出流

OutputStream outputStream = editor.newOutputStream(0);

//url 也就是 下载图片的地址

//outputStream 的作用在于,

//从网络下载图片的时候,图片通过该输出流写到文件系统,

//也就是说,图片下载到了磁盘缓存中。

if (downloadUrlToStream(url, outputStream)) {

editor.commit();

} else {

//释放编辑锁

editor.abort();

}

diskLruCache.flush();

}

} catch (Exception e) {

e.printStackTrace();

}

}

}.start();

/**

* 将key进行加密

*

* @param key

* @return

*/

public String hashKeyForDisk(String key) {

String cacheKey;

try {

final MessageDigest mDigest = MessageDigest.getInstance("MD5");

mDigest.update(key.getBytes());

cacheKey = bytesToHexString(mDigest.digest());

} catch (NoSuchAlgorithmException e) {

cacheKey = String.valueOf(key.hashCode());

}

return cacheKey;

}

private String bytesToHexString(byte[] bytes) {

StringBuilder sb = new StringBuilder();

for (int i = 0; i < bytes.length; i++) {

String hex = Integer.toHexString(0xFF & bytes[i]);

if (hex.length() == 1) {

sb.append('0');

}

sb.append(hex);

}

return sb.toString();

}

我们首先初始化 DiskLruCache.Editor editor对象,把图片的url经过MD5加密,然后作为缓存图片的key。这里为什么不直接用url作为key而要进行md5加密呢。因为url中,可能存在一些特殊字符,这样一来可能在命名文件的时候不合法。 md5加密之后的字符是唯一的,并且都是0-F的字符。然后创建OutputStream outputStream对象

OutputStream outputStream = editor.newOutputStream(0);

下载图片之后就是通过该输出流进行写入文件,也就是说,把下载下来的图片写入到缓存目录中。

//也就是说,图片下载到了磁盘缓存中。

if (downloadUrlToStream(url, outputStream)) {

editor.commit();

} else {

//释放编辑锁

editor.abort();

}

diskLruCache.flush();

下载成功后调用 editor.commit();提交即可。

我们具体看下下载图片的方法

/**

* 从网络中下载图片,并写到缓存中

*

* @param urlString

* @param outputStream

* @return

*/

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;

}

我们看到下载的图片写到 OutputStream outputStream中,也就是写到了缓存中。这样一来就把图片写到了缓存中了。

我们看下缓存图片的目录:

我们看到这里有一个journal文件和一个名字很长的文件,名字很长的文件,就是我们的缓存文件了,因为是经过md5加密后的字符串。

读取缓存

接下里我们介绍如何读取缓存文件。

DiskLruCache.Snapshot snapshot = diskLruCache.get(hashKeyForDisk(url));

if(snapshot!=null){

InputStream inputStream = snapshot.getInputStream(0);

Bitmap bitmap = BitmapFactory.decodeStream(inputStream);

//如果不为空,则直接展示缓存中的bitmap

imageView.setImageBitmap(bitmap);

}

这里为什么是getInputStream(0);呢。因为我们上面定义了一个key对应一个数据,所以只获取第0个即可

我们看下运行的效果图:

移除缓存

调用remove方法,移除指定的数据。

public synchronized boolean remove(String key) throws IOException

其他api

1.flush()

用于将内存中的操作记录同步到日志文件,也就是sdcard中的journal文件。DiskLruCache正常工作就要依赖该文件中的内容。但是没必要每次写入缓存操作的时候都调用一次,一般在Activity的onPause方法中调用一次即可。

2.delete()

删除所有的缓存

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

android强制缓存写磁盘,Android缓存之DiskLruCache磁盘缓存的使用相关推荐

  1. android强制停止运行,如何在android中单击强制停止时运行服务

    单击强制停止时我遇到了运行服务的问题,当我重启我的手机时应该调用服务.我已经按照一些例子但我无法完成任务.任何人都可以指导我完成任务. 需要: 1.Service should run when fo ...

  2. android 强制锁屏代码,Android 监听锁屏、解锁、开屏 功能代码

    1.首先定义 ScreenListener package com.app.lib; import android.content.BroadcastReceiver; import android. ...

  3. android强制弹广告,清除Android游戏强制插入的广告

    在很多论坛和第三方市场都能下载到免费的Android游戏,但绝大多数游戏运行中都会强行弹出插入的广告条.那么,如何才能"干掉"这些扰人的广告呢? 所谓"天下没有免费的午餐 ...

  4. android 输入法 候选,写一个Android输入法02——候选窗、转换

    上一篇介绍了完成Android输入法的最小化步骤,它只能将按键对应的字符上屏.一般的东亚语言都有一个转换的过程,比如汉语输入拼音,需要由拼音转成汉字再上屏.本文将在前文基础上加入完成转换过程所必需的候 ...

  5. android开发怎么写图片,Android开发实践:自己动手编写图片剪裁应用(2)

    上篇文章主要介绍了我开源在Github上的图片剪裁库(ImageCropper)的基本特性和用法,从本文开始,慢慢介绍一些开发图片剪裁应用中涉及的知识点和技术. 其实Android系统本身也提供了图片 ...

  6. Android 系统(181 )----Android中各类.mk文件的编写

    Android中各类.mk文件的编写 LOCAL_PATH:= $(call my-dir) include $(CLEAR_VARS) LOCAL_STATIC_JAVA_LIBRARIES := ...

  7. android强制缓存写磁盘,优雅的构建 Android 项目之磁盘缓存(DiskLruCache)

    Android 的缓存技术 一个优秀的应用首先它的用户体验是优秀的,在 Android 应用中恰当的使用缓存技术不仅可以缓解服务器压力还可以优化用户的使用体验,减少用户流量的使用.在 Android ...

  8. Android RxJava操作符的学习---组合合并操作符---从磁盘或内存缓存中获取缓存数据

    1. 需求场景 2. 功能说明 对于从磁盘 / 内存缓存中 获取缓存数据 的功能逻辑如下: 3. 具体实现 详细请看代码注释 // 该2变量用于模拟内存缓存 & 磁盘缓存中的数据String ...

  9. android 自动清理缓存图片吗,Android中Glide获取缓存大小并清除缓存图片

    清除Glide缓存 Glide自带清除缓存的功能,分别对应Glide.get(context).clearDiskCache();(清除磁盘缓存)与Glide.get(context).clearMe ...

最新文章

  1. VS Code 调试 OneFlow
  2. React Native —— App
  3. 图解WindowsXP修改MAC地址
  4. 深入分析线程池的实现原理
  5. wampserver2.5安装 redis缓存,igbinary, phalcon框架
  6. oracle各版本手动升级任我行-升级矩阵
  7. abap alv新增行数据_ALV DMEO 09:REUSE_ALV_GRID_DISPLAY 使用HTML 居中 颜色大小 加粗 斜体 超链接 控制...
  8. 用过哪些Map类,都有什么区别,HashMap是线程安全的吗,并发下使用的Map是什么,他们内部原理分别是什么,比如存储方式,hashcode,扩容,默认容量等。
  9. 【CodeForces - 735B】Urbanization (找规律,思维)
  10. flask框架+mysql数据库并与前台数据交互
  11. jdk 安装_Linux安装JDK操作手册
  12. [No000048]程序员的成长过程中,有哪些阶段?
  13. Kubernetes详解(七)——Service对象部署和应用
  14. Ext4,Ext3的特点和区别
  15. 荔枝软件如何测试声音,荔枝如何测自己的声音 荔枝测自己的声音方法
  16. java 获取身份证前两位对应省份
  17. WTL入门(五) 自定义控件
  18. 你真的了解人工智能吗?——聊聊AI的碰壁和冬天
  19. 联想TinkPad S3-490 后盖拆机教程
  20. pcode.linq

热门文章

  1. 12月21日云栖精选夜读 | 推荐:一款分布式的对象存储服务
  2. java的 oo_Java OO 知识总结
  3. 蓝桥杯双阶乘 (python)
  4. unity 音乐节奏游戏_使用您当地音乐收藏的最佳节奏游戏
  5. 2022考研笔记-数学(高等数学预备知识)
  6. VScode 4 括号颜色分级插件(Bracket Pair Colorizer)
  7. 2022-2028年中国零售业信息化行业市场深度评估及投资机会预测报告
  8. carsim输入模块设置问题
  9. 数字图像处理第十章 图像分割
  10. AWS CloudFormation