android读取大图片并缓存
UI组件方面采用Gallery+ImageSwitcher组合,这里略过,详情参见google
Android API。 相册图片预取缓存策略是内存缓存(硬引用LruCache、软引用SoftReference<Bitmap>)、外部文件缓存(context.getCachedDir()),缓存中取不到的情况下再向服务端请求下载图片。同时缓存三张图片(当前预览的这张,前一张以及后一张)。 1.内存缓存 //需要导入外部jar文件
android-support-v4.jar
import
android.support.v4.util.LruCache;
//开辟8M硬缓存空间
private final
int hardCachedSize = 8*1024*1024;
//hard
cache
private final LruCache<String, Bitmap> sHardBitmapCache =
new LruCache<String,
Bitmap>(hardCachedSize){
@Override
public int
sizeOf(String key, Bitmap value){
return
value.getRowBytes() *
value.getHeight();
}
@Override
protected
void entryRemoved(boolean evicted, String key, Bitmap oldValue, Bitmap
newValue){
Log.v("tag", "hard cache is full , push to soft
cache");
//硬引用缓存区满,将一个最不经常使用的oldvalue推入到软引用缓存区
sSoftBitmapCahe.put(key,
new
SoftReference<Bitmap>(oldValue));
}
}
//软引用
private
static final int SOFT_CACHE_CAPACITY = 40;
private final static
LinkedHashMap<String, SoftReference<Bitmap>> sSoftBitmapCache =
new LinkedHashMao<String,
SoftReference<Bitmap>>(SOFT_CACHE_CAPACITY, 0.75f,
true){
@Override
public
SoftReference<Bitmap> put(String key, SoftReference<Bitmap>
value){
return super.input(key,
value);
}
@Override
protected
boolean removeEldestEntry(LinkedHashMap.Entry<Stirng,
SoftReference<Bitmap>> eldest){
if(size() >
SOFT_CACHE_CAPACITY){
Log.v("tag", "Soft Reference
limit , purge one");
return
true;
}
return
false;
}
}
//缓存bitmap
public boolean
putBitmap(String key, Bitmap bitmap){
if(bitmap !=
null){
synchronized(sHardBitmapCache){
sHardBitmapCache.put(key,
bitmap);
}
return
true;
}
return
false;
}
//从缓存中获取bitmap
public Bitmap
getBitmap(String
key){
synchronized(sHardBitmapCache){
final
Bitmap bitmap = sHardBitmapCache.get(key);
if(bitmap !=
null)
return
bitmap;
}
//硬引用缓存区间中读取失败,从软引用缓存区间读取
synchronized(sSoftBitmapCache){
SoftReference<Bitmap>
bitmapReference =
sSoftBtimapCache.get(key);
if(bitmapReference !=
null){
final Bitmap bitmap2 =
bitmapReference.get();
if(bitmap2 !=
null)
return
bitmap2;
else{
Log.v("tag",
"soft reference
已经被回收");
sSoftBitmapCache.remove(key);
}
}
}
return
null;
} 2.外部文件缓存 private
File mCacheDir = context.getCacheDir();
private static final int
MAX_CACHE_SIZE = 20 * 1024 * 1024; //20M
private final
LruCache<String, Long> sFileCache = new LruCache<String,
Long>(MAX_CACHE_SIZE){
@Override
public int
sizeOf(String key, Long value){
return
value.intValue();
}
@Override
protected
void entryRemoved(boolean evicted, String key, Long oldValue, Long
newValue){
File file =
getFile(key);
if(file !=
null)
file.delete();
}
}
private
File getFile(String fileName) throws FileNotFoundException {
File
file = new File(mCacheDir, fileName);
if(!file.exists() ||
!file.isFile())
throw new
FileNotFoundException("文件不存在或有同名文件夹");
return
file;
}
//缓存bitmap到外部存储
public boolean
putBitmap(String key, Bitmap bitmap){
File file =
getFile(key);
if(file != null){
Log.v("tag",
"文件已经存在");
return
true;
}
FileOutputStream fos =
getOutputStream(key);
boolean saved =
bitmap.compress(CompressFormat.JPEG, 100,
fos);
fos.flush();
fos.close();
if(saved){
synchronized(sFileCache){
sFileCache.put(key,
getFile(key).length());
}
return true;
}
return
false;
}
//根据key获取OutputStream
private
FileOutputStream getOutputStream(String key){
if(mCacheDir ==
null)
return null;
FileOutputStream fos =
new FileOutputStream(mCacheDir.getAbsolutePath() + File.separator +
key);
return fos;
}
//获取bitmap
private
static BitmapFactory.Options sBitmapOptions;
static
{
sBitmapOptions = new
BitmapFactory.Options();
sBitmapOptions.inPurgeable=true;
//bitmap can be purged to disk
}
public Bitmap
getBitmap(String key){
File bitmapFile =
getFile(key);
if(bitmapFile != null){
Bitmap
bitmap = BitmapFactory.decodeStream(new FileInputStream(bitmapFile), null,
sBitmapOptions);
if(bitmap !=
null){
//重新将其缓存至硬引用中
...
}
}
} 3.从服务端下载图片 private
static String generateKey(String fileId, int width, int height)
{
String ret = fileId + "_" + Integer.toString(width)
+ "x" + Integer.toString(height);
return
ret;
}
String key = generateKey(...)即可生成唯一的key值 下载成功后调用1内存缓存的putBitmap()函数,缓存图片。 在外部文件缓存中也写入一份,调用2的putBitmap()函数. 4.预览图片的流程 1)
如果预览的图片在内存缓存区中,直接调用1的getBitmap()函数,获取bitmap数据(先在硬引用缓存区查找匹配,若硬引用区匹配失败,再去软引用区匹配) 2)
如果从内存缓存区读取失败,再从外部文件缓存中读取,调用2的getBitmap()函数 3)
如果从外部文件缓存中读取失败,则从服务端下载该图片,过程3. 5.生成key值
android读取大图片并缓存的更多相关文章
- android对大图片的缓存处理
废话不多说,直接上代码 package com.huge.emj.common.util; import java.io.File; import java.io.FileInputStream; i ...
- Android异步下载图片并且缓存图片到本地
Android异步下载图片并且缓存图片到本地 在Android开发中我们经常有这样的需求,从服务器上下载xml或者JSON类型的数据,其中包括一些图片资源,本demo模拟了这个需求,从网络上加载XML ...
- android读取远程图片案例
关键代码:Bitmap bitmap=BitmapFactory.decodeByteArray(data, 0, data.length);imageview.setImageBitmap(bitm ...
- Android读取url图片保存及文件读取
参考: 1.http://blog.csdn.net/ameyume/article/details/6528205 2.http://blog.sina.com.cn/s/blog_85b3a161 ...
- Android 显示大图片
主要的代码如下: BitmapFactory.Options options = new BitmapFactory.Options(); //图片解析配置 options.inJustDecodeB ...
- android 应用开发对大图片的处理
一,下载 android下载大图片(例如微博长图片)会出现OOM down掉问题 解决这个问题的办法是下载图片时先得到图片的宽度和高度,如果超出规定限制则对图片进行缩放 关键参数 1. BitmapF ...
- Android调用系统相机、自己定义相机、处理大图片
Android调用系统相机和自己定义相机实例 本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,而且因为涉及到要把拍到的照片显示出来,该样例也会涉及到Android载入大图片时候的处 ...
- Android Camera开发系列(上)——Camera的基本调用与实现拍照功能以及获取拍照图片加载大图片
Android Camera开发系列(上)--Camera的基本调用与实现拍照功能以及获取拍照图片加载大图片 最近也是在搞个破相机,兼容性那叫一个不忍直视啊,于是自己翻阅了一些基本的资料,自己实现了一 ...
- java快速获取大图片的分辨率(大图片格式JPG,tiff ,eg)
问题描述:怎样快速获取一个20MB图片的分辨率? 程序代码: package test; import java.awt.Dimension; import java.awt.image.Buffer ...
随机推荐
- Vue2.0 v-for 中 :key 的作用
- Requests: 让 HTTP 服务人类
requests 2.18.1文档 requests流式post文件 Calling SOAP Web service using requests module of
- TouchEvent: dispatchTouchEvent(), onTouch() , onTouchEvent(), requestDisallowInterceptTouchEvent() 方法中的一些细节
Q: onTouchListener与onClickListener谁先执行?A: onTouchListener中的内容先执行.若其中的onTouch方法返回true,代表该事件已被onTouch消 ...
- SQL Server 2008 R2:error 26 开启远程连接详解
远程连接sql server 2008 数据库,出现下面的错误: <--在与 SQL Server 建立连接时出现与网络相关的或特定于实例的错误. 未找到或无法访问服务器.请验证实例名称是 ...
- 【Android】Handler详解
Handler的定义 主要接受子线程发送的数据, 并用此数据配合主线程更新UI. 解释: 当应用程序启动时,Android首先会开启一个主线程 (也就是UI线程) , 主线程为管理界面中的UI控件,进 ...
- 【嵌入式】使用Cross Toolchain构建交叉工具链
Preface 前面编译linux内核的时候,用各种cross版本都不行啊,真是纠结,于是就想着自己也要会编译交叉工具的方法,然后各种尝试,各种问题啊,最后还是没解决(还有其它事情),步骤我都走熟了, ...
- 【Bayesian】贝叶斯决策方法(Bayesian Decision Method)
已知某条件概率,如何得到两个事件交换后的概率,也就是在已知P(A|B)的情况下如何求得P(B|A).这里先解释什么是条件概率: 表示事件B已经发生的前提下,事件A发生的概率,叫做事件B发生下事件A的条 ...
- 【linux】dpkg info修复及dpkg: warning: files list file for package
mv /var/lib/dpkg/info /var/lib/dpkg/info.bak //现将info文件夹更名 sudo mkdir /var/lib/dpkg/info //再新建一个新的in ...
- 2. K-Means的优化
1. K-Means原理解析 2. K-Means的优化 3. sklearn的K-Means的使用 4. K-Means和K-Means++实现 1. 前言 上一篇博文K-Means原理解析简单清晰 ...
- [转]关于重定向RedirectAttributes的用法
刚才做项目的时候看到一篇写的很不错的博客,解决我之前对于RedirectAttributes的困惑,也给大家推荐下~ 原文地址https://zhuanlan.zhihu.com/p/21353217 ...