Cache【硬盘缓存工具类(包含内存缓存LruCache和磁盘缓存DiskLruCache)】
版权声明:本文为HaiyuKing原创文章,转载请注明出处!
前言
内存缓存LruCache和磁盘缓存DiskLruCache的封装类,主要用于图片缓存。
效果图

代码分析
内存缓存LruCache和磁盘缓存DiskLruCache根据实际情况进行缓存,有时候需要两者都用到,有时候只用磁盘缓存即可。
使用步骤
一、项目组织结构图

注意事项:
1、 导入类文件后需要change包名以及重新import R文件路径
2、 Values目录下的文件(strings.xml、dimens.xml、colors.xml等),如果项目中存在,则复制里面的内容,不要整个覆盖
二、导入步骤
将Cache、DiskLruCache文件复制到项目中

在AndroidManifest.xml中添加权限
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.why.project.cachedemo"> <!-- =====================Cache【硬盘缓存工具类(包含内存缓存LruCache和磁盘缓存DiskLruCache)】======================= -->
<!-- 允许程序打开网络套接字 -->
<uses-permission android:name="android.permission.INTERNET"/>
<!-- 允许程序访问有关GSM网络信息 -->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<!-- 允许程序访问Wi-Fi网络状态信息 -->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<!-- 在SD卡中创建与删除文件权限 -->
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<!-- 允许程序读取外部存储文件 -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<!-- 允许程序写入外部存储,如SD卡上写文件 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN"/> <category android:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
</application> </manifest>
添加运行时权限的处理(本demo中采用的是修改targetSDKVersion=22)
三、使用方法
package com.why.project.cachedemo; import android.content.res.Resources;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.drawable.BitmapDrawable;
import android.os.AsyncTask;
import android.os.Build;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.TextView; import com.why.project.cachedemo.utils.cache.Cache; import java.net.URL; public class MainActivity extends AppCompatActivity { private Button btn_save;
private Button btn_show;
private Button btn_getsize;
private Button btn_remove; private TextView tv_size;
private ImageView img_show; private GetImgToSaveCacheTask getImgToSaveCacheTask;//获取图片并保存到缓存的异步请求类 /**内存、磁盘缓存*/
private Cache mCache;
String imgUrl = "http://img.ithome.com/newsuploadfiles/2014/12/20141223_115629_592.jpg"; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); /**必须在使用之前初始化*/
mCache = new Cache(this);
mCache.initMemoryCache();
mCache.initDiskCache(); initViews();
initEvents();
} @Override
public void onPause() {
super.onPause(); //cancel方法只是将对应的AsyncTask标记为cancel状态,并不是真正的取消线程的执行,在Java中并不能粗暴的停止线程,只能等线程执行完之后做后面的操作
if(getImgToSaveCacheTask != null && getImgToSaveCacheTask.getStatus() == AsyncTask.Status.RUNNING){
getImgToSaveCacheTask.cancel(true);
} mCache.clearCache();
System.gc();
} private void initViews() {
btn_save = (Button) findViewById(R.id.btn_save);
btn_show = (Button) findViewById(R.id.btn_show);
btn_getsize = (Button) findViewById(R.id.btn_getsize);
btn_remove = (Button) findViewById(R.id.btn_remove); tv_size = (TextView) findViewById(R.id.tv_size);
img_show = (ImageView) findViewById(R.id.img_show);
} private void initEvents() {
btn_save.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) { saveToCache(imgUrl);
}
}); btn_show.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
showImg(imgUrl);
}
}); btn_getsize.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String cacheSize = mCache.getSize();
tv_size.setText(tv_size.getText() + cacheSize);
}
}); btn_remove.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mCache.removeFromCache(mCache.hashKeyForDisk(imgUrl));
}
});
} /**显示图片*/
private void showImg(String imgUrl) {
String cacheKey = mCache.hashKeyForDisk(imgUrl);
Bitmap bitmap = mCache.getBitmapFromCache(cacheKey);//从缓存中获取图片bitmap
if(bitmap != null){
BitmapDrawable drawable = null;
Resources resources = getResources();
drawable = new BitmapDrawable(resources, bitmap);
//设置背景图片
if(Build.VERSION.SDK_INT >= 16) {
img_show.setBackground(drawable);
}else{
img_show.setBackgroundDrawable(drawable);
}
}
} /**保存图标到缓存中*/
private void saveToCache(String imgUrl){ //如果已经下载过来,那么不需要重新下载
String cacheKey = mCache.hashKeyForDisk(imgUrl);
Bitmap bitmap = mCache.getBitmapFromCache(cacheKey);//从缓存中获取图片bitmap
if(bitmap == null){
//判断是否有网络,此处注释掉
//if (HttpUtil.isNetworkAvailable(MainActivity.this)) {
//执行异步任务
GetImgToSaveCacheTask getImgTask = new GetImgToSaveCacheTask();
getImgTask.execute(imgUrl);
/*} else {
this.showShortToast(getResources().getString(R.string.httpError));
}*/
}
} /**
* 保存图标异步请求类
*/
public class GetImgToSaveCacheTask extends AsyncTask<String, Void, String> { @Override
protected void onPreExecute() {
//showProgressDialog("");//显示加载对话框
}
@Override
protected String doInBackground(String... params) {
String data = "";
if(! isCancelled()){
try {
if(params[0].equals("")){
data = "";
}else{
//下载图片并缓存
BitmapFactory.Options opts = new BitmapFactory.Options(); String cacheKey = mCache.hashKeyForDisk(params[0]);
Bitmap bitmap = null;
synchronized(mCache.mDiskCacheLock){
if(mCache.getBitmapFromCache(cacheKey) == null) {
//如果缓存中无图片,则添加到磁盘缓存和内存缓存中
bitmap = BitmapFactory.decodeStream(new URL(params[0]).openConnection().getInputStream(), null, opts);
mCache.addBitmapToMemoryCache(cacheKey, bitmap);
mCache.addBitmapToDiskCache(cacheKey, bitmap);
} else if(mCache.mMemoryCache.get(cacheKey) == null) {
//如果内存缓存中无图片,则磁盘缓存中添加到内存缓存中
bitmap = mCache.getBitmapFromDiskCache(cacheKey);
mCache.addBitmapToMemoryCache(cacheKey, bitmap);
} else if(mCache.getBitmapFromDiskCache(cacheKey) == null) {
//如果磁盘缓存中无图片,则从内存缓存中添加到磁盘缓存中
bitmap = mCache.getBitmapFromMemCache(cacheKey);
mCache.addBitmapToDiskCache(cacheKey, bitmap);
}
}
data = "success";
}
} catch (Exception e) {
e.printStackTrace();
}
}
return data;
}
@Override
protected void onPostExecute(String result) {
super.onPostExecute(result);
if(isCancelled()){
return;
}
try {
if (result != null && !"".equals(result)){
//缓存成功,此处可以进行其他处理,比如显示图片
}
} catch (Exception e) {
e.printStackTrace();
} finally {
if(! isCancelled()){
//dismissProgressDialog();//隐藏加载对话框
}
}
}
}
}
注意:
- 缓存图片建议使用异步请求,demo中使用的是AsyncTask方式;
- 异步请求缓存图片之前需要先进行网络判断,请参考HttpUtil 【判断网络连接的封装类】
缓存目录示意图:

混淆配置
无
参考资料
Android DiskLruCache完全解析,硬盘缓存的最佳方案
项目demo下载地址
https://github.com/haiyuKing/CacheDemo
Cache【硬盘缓存工具类(包含内存缓存LruCache和磁盘缓存DiskLruCache)】的更多相关文章
- 分享基于MemoryCache(内存缓存)的缓存工具类,C# B/S 、C/S项目均可以使用!
using System; using System.Collections.Generic; using System.Linq; using System.Runtime.Caching; usi ...
- Go/Python/Erlang编程语言对比分析及示例 基于RabbitMQ.Client组件实现RabbitMQ可复用的 ConnectionPool(连接池) 封装一个基于NLog+NLog.Mongo的日志记录工具类LogUtil 分享基于MemoryCache(内存缓存)的缓存工具类,C# B/S 、C/S项目均可以使用!
Go/Python/Erlang编程语言对比分析及示例 本文主要是介绍Go,从语言对比分析的角度切入.之所以选择与Python.Erlang对比,是因为做为高级语言,它们语言特性上有较大的相似性, ...
- php 缓存工具类 实现网页缓存
php 缓存工具类 实现网页缓存 php程序在抵抗大流量访问的时候动态网站往往都是难以招架,所以要引入缓存机制,一般情况下有两种类型缓存 一.文件缓存 二.数据查询结果缓存,使用内存来实现高速缓存 本 ...
- redis缓存工具类
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis ...
- redis缓存工具类,提供序列化接口
1.序列化工具类 package com.qicheshetuan.backend.util; import java.io.ByteArrayInputStream; import java.io. ...
- CookieUtils-浏览器缓存工具类
package cn.yonyong.myproject.commons.utils; import javax.servlet.http.Cookie; import javax.servlet.h ...
- 缓存工具类CacheHelper
代码: using System; using System.Collections.Generic; using System.Linq; using System.Text; using Syst ...
- 基于spring的redisTemplate的缓存工具类
pom.xml文件添加 <!-- config redis data and client jar --><dependency> <groupId>org.spr ...
- System.Web.Caching.Cache缓存帮助类
/// <summary> /// 缓存帮助类 /// </summary> public class CacheHelper { /// <summary> // ...
随机推荐
- 【转】用信鸽来解释 HTTPS
一.引文出处 注:分享交流技术.本文摘自开源中国翻译,翻译者:JonnHuang,句号句号 译文原网址:https://www.oschina.net/translate/https-explaine ...
- Python ymal 模块和configparser
ymal : 是一种config文件 # !/user/bin/python # -*- coding: utf-8 -*- import configparser # 生成一个config文件 (当 ...
- plugin.go 源码阅读
, nil) } if c.client != nil { c.client.Close() } ...
- MySQL 开发实践
最近研发的项目对DB依赖比较重,梳理了这段时间使用MySQL遇到的8个比较具有代表性的问题,答案也比较偏自己的开发实践,没有DBA专业和深入,有出入的请使劲拍砖!- MySQL读写性能是多少,有哪些性 ...
- php与html实现交互的基本操作
今天我们来实现php与html页面注册和登录的效果.中国有句古话叫: 第一步:我们来了解一些php的基本格式. <?php php代码 ?> 第二步:了解php与js的一些基本区别 我们在 ...
- TensorFlow TensorBoard使用
摘要: 1.代码例子 2.主要功能内容: 1.代码例子 <TensorFlow实战>使用MLP处理Mnist数据集并TensorBoard上显示 2.主要功能 执行TensorBoard程 ...
- 对于单页应用中如何监听 URL 变化的思考
周末开发了一个在 GitHub 中给 repo 增加自定义备注的 chrome 扩展. 开发这个扩展的原因是我在 GitHub 中所 star 的项目实在太多了(截止目前 671 个),有的项目过个几 ...
- SpringBoot之旅第四篇-web开发
一.引言 有了自动配置,springboot使web开发变得简单,这个在springboot之旅中的第一篇中就有体现,实际的开发中当然不会这么简单,很多时候我们都需要自己去定制一些东西.web开发的东 ...
- 从mysql中拿到的数据构造为列表
最近测试接口遇到一个问题,用python2.7从mysql中取到的数据是元祖类型的,元祖内部的元素也是一个元祖(并且部分元素的编码格式是unicode的): 类似这样: ((10144, u''), ...
- zookeeper源码 — 一、单机启动
zookeeper一般使用命令工具启动,启动主要就是初始化所有组件,让server可以接收并处理来自client的请求.本文主要结构: main入口 配置解析 组件启动 main入口 我们一般使用命令 ...