- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章Android开发之ImageLoader本地缓存由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
ImageLoader是一个图片缓存的开源库,提供了强大的图片缓存机制,很多开发者都在使用,今天给大家介绍Android开发之ImageLoader本地缓存,具体内容如下所示:
本地缓存在缓存文件时对文件名称的修改提供了两种方式,每一种方式对应了一个Java类 。
1) HashCodeFileNameGenerator ,该类负责获取文件名称的hashcode然后转换成字符串.
2) Md5FileNameGenerator ,该类把源文件的名称同过md5加密后保存.
两个类都继承了FileNameGenerator接口 。
在DefaultConfigurationFactory类中提供了一个工厂方法createFileNameGenerator,该方法返回了一个默认的FileNameGenerator对象:HashCodeFileNameGenerator. 。
1
2
3
|
public
static
FileNameGenerator createFileNameGenerator() {
return
new
HashCodeFileNameGenerator();
}
|
实现 。
首先定义了DiscCacheAware接口,该接口提供了如下方法 。
1
2
3
4
5
6
7
|
File getFileDectory() 返回磁盘缓存的根目录
File get(String imageUri) 根据uri从缓存中获取图片
boolean
save(String imageUri,InputStream iamgeStream,IoUtils.CopyListener listener) 把图片保存在磁盘缓存上
boolean
save(String imageUri,Bitmap bitmap) 保存bitmap对象到磁盘缓存上
boolean
remove(imageUri) 根据imageUri删除文件
void
close() 关闭磁盘缓存,释放资源
void
clear() 清空磁盘缓存
|
然后定义了另外一个没方法的接口DiskCache,该接口只是简单的继承了DiscCacheAware接口.
BaseDiscCache实现了DiskCache,该类是个抽象类,该类定义了磁盘缓冲区的以下的属性:
1) 默认的缓存大小为32k 。
2) 默认压缩后的图片格式为PNG(作为Bitmap的compress方法的第一个参数) 。
3) 默认压缩后图片显示的质量为100,也就是压缩率为0,不进行压缩(作为compress的第二个参数) 。
提供了修改压缩图片格式和压缩率以及修改缓存大小的set方法。同时该类还封装了以下三个属性 。
1
2
3
|
protected
final
File cacheDir;
//缓存文件的保存Directory
protected
final
File reserveCacheDir;
//后备缓存的Diectory,当cacheDir不存在的情况下就是用reserveCahceDir后备缓存
protected
final
FileNameGenerator fileNameGenerator;
//文件名名称生成器
|
构造函数 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
public
BaseDiscCache(File cacheDir) {
this
(cacheDir,
null
);
}
public
BaseDiscCache(File cacheDir, File reserveCacheDir) {
this
(cacheDir, reserveCacheDir, DefaultConfigurationFactory.createFileNameGenerator());
}
public
BaseDiscCache(File cacheDir, File reserveCacheDir, FileNameGenerator fileNameGenerator) {
if
(cacheDir ==
null
) {
throw
new
IllegalArgumentException(
"cacheDir"
+ ERROR_ARG_NULL);
}
if
(fileNameGenerator ==
null
) {
throw
new
IllegalArgumentException(
"fileNameGenerator"
+ ERROR_ARG_NULL);
}
this
.cacheDir = cacheDir;
this
.reserveCacheDir = reserveCacheDir;
this
.fileNameGenerator = fileNameGenerator;
}
|
1) 只有一个参数的构造函数只初始化了cacheDir,没有用到后备缓存,且是以HashCodeFileNameGenerator来生成目标文件的文件名.
2) 两个参数的构造器除了cacheDir和HashCodefileNameGenerator外,也可以初始化后备缓存 。
3) 三个参数的构造器要求必须初始化cacheDir并且必须初始化filenNameGenerator否则就报异常 。
get(String imageUri) 。
1
2
3
4
5
6
7
8
9
10
|
protected
File getFile(String imageUri) {
String fileName = fileNameGenerator.generate(imageUri);
File dir = cacheDir;
if
(!cacheDir.exists() && !cacheDir.mkdirs()) {
if
(reserveCacheDir !=
null
&& (reserveCacheDir.exists() || reserveCacheDir.mkdirs())) {
dir = reserveCacheDir;
}
}
return
new
File(dir, fileName);
}
|
save(String imageUri, Bitmap bitmap) 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
public
boolean
save(String imageUri, Bitmap bitmap)
throws
IOException {
//获取imageUri的File对象,该对象封装了缓存路径和图片保存后的名称
File imageFile = getFile(imageUri);
//获取临时保存文件的tmpFile对象
File tmpFile =
new
File(imageFile.getAbsolutePath() + TEMP_IMAGE_POSTFIX);
OutputStream os =
new
BufferedOutputStream(
new
FileOutputStream(tmpFile), bufferSize);
boolean
savedSuccessfully =
false
;
try
{
//调用compress把bitMap压缩到tempFile中
savedSuccessfully = bitmap.compress(compressFormat, compressQuality, os);
}
finally
{
IoUtils.closeSilently(os);
//如果保存成功并且tempFile的文件没有成功移动到imageFile的话,就删除temFile
if
(savedSuccessfully && !tmpFile.renameTo(imageFile)) {
savedSuccessfully =
false
;
}
if
(!savedSuccessfully) {
tmpFile.delete();
}
}
//对bitmap进行垃圾回收
bitmap.recycle();
return
savedSuccessfully;
}
|
BaseDiscCache有两个扩展类,一个是 不限制缓存大小的 UnlimitedDiscCache 和 限制缓存时间的LimitedAgeDiscCache, 其中UnlimitedDiscCache很简单它只是简单的继承了BaseDiscCache并未对BaseDiscCache做任何扩展.
LimitedAgeDiscCache 该类实现了在缓存中删除被加载超过规定时间的文件: 满足以下条件的时候就从缓存中删除文件:系统当前时间-文件的最新修改时间 > maxFileAge 。
LimitedAgeDiscCache 。
该类提供了两个属性:
1. maxFileAge(long)设置加载的超时的最大时间,改时间在构造器冲初始化,一经初始化就不能改变(设定文件存活的最长时间,当超过这个值,就删除该文件) 。
2. loadingDates (Map<File,long>),该属性是一个map类型的对象,key保存的要缓存的图片文件,而value保存的是调用save方法是系统的当前时间,具体向loadingDates填充数据是在下面的rememberUsage方法中实现的,该方法在类中两个save方法中调用,首先调用父类的save方法,然后在调用此方法 。
1
2
3
4
5
6
|
private
void
rememberUsage(String imageUri) {
File file = getFile(imageUri);
long
currentTime = System.currentTimeMillis();
file.setLastModified(currentTime);
loadingDates.put(file, currentTime);
}
|
从缓存中获取数据的方法为get(String imageUri)该类是重写BaseDiscDache方法,该方法从loadingDates中获取imageUri所代表的图片的最新更新时间loadingDate, 然后拿当前时间和loadingDate做差,如果差值大于maxFileAge也就是说查过了加载的最大时间,就删除该imageUri所代表的file,并从loadingDates中的数据,当然如果map中没有imageUri就不会涉及到超时的问题,此时就把image放入map中去 ,具体的实现如下 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
@Override
public
File get(String imageUri) {
File file =
super
.get(imageUri);
if
(file !=
null
&& file.exists()) {
boolean
cached;
Long loadingDate = loadingDates.get(file);
if
(loadingDate ==
null
) {
cached =
false
;
loadingDate = file.lastModified();
}
else
{
cached =
true
;
}
if
(System.currentTimeMillis() - loadingDate > maxFileAge) {
file.delete();
loadingDates.remove(file);
}
else
if
(!cached) {
loadingDates.put(file, loadingDate);
}
}
return
file;
}
|
最后此篇关于Android开发之ImageLoader本地缓存的文章就讲到这里了,如果你想了解更多关于Android开发之ImageLoader本地缓存的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我需要一些说明。我可以直接写入 /dev/port 以直接访问并行端口并且它工作正常(我可以打开插入端口连接器的 LED)。但是,我想我可以用 /dev/mem 做同样的事情? (http://tld
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我使用 Visual C++ 和 Win32 API 学习了 Windows 编程。如今,似乎大多数应用程序都是使用 C# 在 .NET 中开发的。我知道大多数时候 native 代码和托管代码之间没
请耐心等待。我正在制作一个 java 控制台,类似于此处找到的 DragonConsole https://code.google.com/p/dragonconsole/ 。一切都按计划进行,但我想
关闭。这个问题需要更多 focused .它目前不接受答案。 想要改进这个问题吗? 更新问题,使其只关注一个问题 editing this post . 关闭5年前。 Improve this que
Django 的开发服务器表现得很奇怪。访问它的浏览器在加载时卡住,任何退出它的尝试都不起作用。当我点击 control c看似相当,但实际上仍在运行。让它退出的唯一方法是重新启动我的电脑,这很令人沮
我正在使用 Flash Develop,并且创建了一个 ActionScript 3.0 项目。它启动并读取一个 xml 文件,其中包含图像的 url。我已将 url 保留在与 swf 相同的文件夹中
是否可以根据其 website 上提供的规范开发 AUTOSAR BSW 堆栈(例如用于 CAN 通信)?不购买任何昂贵的供应商工具?可以遵循哪些步骤?我被要求探索这种可能性。 最佳答案 是和否。工具
有人知道如何用音频文件的内容覆盖 iPhone 麦克风吗? 想象一个场景,您正在通话,并且想要播放一些简短的音频让其他人听到。 因此,有必要将麦克风(硬件)置于保持状态,并使用委托(delegate)
我遇到了这个问题,我的应用程序出现 EXC_BAD_ACCESS 错误并卡住/停止。我使用模拟器的“向左旋转”和“向右旋转”选项来模拟方向变化行为。导致此错误的可能原因有哪些?由于我没有获得有关错误的
我有超过 1 台 Mac,我想在所有这些 Mac 上进行开发。我知道我需要在每台机器上同步我的手机,但这是我遇到的最小的问题。看起来我无法在手机上运行应用程序,除了在其中之一上开发的应用程序。 是否有
在手机上测试时,我的应用程序在特定点崩溃。控制台显示此消息 Tue Jan 27 15:47:14 unknown SpringBoard[22] : Application com.myprof.
我有一个案例,我从服务器获取信息。我的应用程序有一个选项卡栏和导航按钮。我希望应用程序显示进度指示器并禁用所有其他控件,以便用户在从服务器提取数据时无法跳转。我怎样才能实现这个目标? 我想到的一种方法
有时,当我尝试“构建”/编译下载的源代码时,我会收到以下警告: ld: warning: directory '/Volumes/Skiiing2/CD/ViewBased/Unknown Path/
我无法在 Apple 文档中找到关于开发和分发配置之间差异的明确解释。我目前正在使用开发配置在我的 iPhone 上进行开发和测试。我打算将该应用程序分发到我的 Beta 测试中,我想知道: 我需要使
我在使用 SharePoint 时遇到的最大挑战之一是它不能很好地适应典型的项目环境,其中至少包含开发和生产环境。我遇到的最多的问题是内容和列表是如此紧密地耦合在一起,以至于如果不在生产环境中执行内容
我失败了fist step让 Eclipse(对我来说是全新的)为 ARM 开发做好准备。 我在 Windows 10 中安装了 Eclipse。我想我应该安装 xpm,但我不知道在哪里输入此命令:
首先,我告诉你-我是编码新手 我正在使用vs代码来学习c++,它不会产生像dev c++或codeblocks这样的调试器。我看了一些视频,其中我们必须编辑json文件,这对于初学者来说非常复杂。有人
我失败了fist step让 Eclipse(对我来说是全新的)为 ARM 开发做好准备。 我在 Windows 10 中安装了 Eclipse。我想我应该安装 xpm,但我不知道在哪里输入此命令:
我开发了一个 Ionic 应用程序(iOS 和 Android 的混合)。我有 Xcode 8.3.3 并购买了一年的 Apple Developer Program 订阅。 我不想测试我的应用并将其
我是一名优秀的程序员,十分优秀!