Android 缓存机制

相册图片预取缓存策略是内存缓存(硬引用LruCache、软引用SoftReference)、外部文件缓存(context.getCachedDir()),缓存中取不到的情况下再向服务端请求下载图片。同时缓存三张图片(当前预览的这张,前一张以及后一张)。

1.内存缓存

[html] view plaincopy

//需要导入外部jar文件 android-support-v4.jar

import android.support.v4.util.LruCache;

//开辟8M硬缓存空间

private final int hardCachedSize = 8*1024*1024;

//hard cache

private final LruCache sHardBitmapCache = new LruCache(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(oldValue));

}

}

//软引用

private static final int SOFT_CACHE_CAPACITY = 40;

private final static LinkedHashMap> sSoftBitmapCache =

new  LinkedHashMao>(SOFT_CACHE_CAPACITY, 0.75f, true){

@Override

public SoftReference put(String key, SoftReference value){

return super.input(key, value);

}

@Override

protected boolean removeEldestEntry(LinkedHashMap.Entry> 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 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.外部文件缓存

[html] view plaincopy

private File mCacheDir = context.getCacheDir();

private static final int MAX_CACHE_SIZE = 20 * 1024 * 1024; //20M

private final LruCache sFileCache = new LruCache(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.从服务端下载图片

下载成功后调用1内存缓存的putBitmap()函数,缓存图片。

在外部文件缓存中也写入一份,调用2的putBitmap()函数.

4.预览图片的流程

1) 如果预览的图片在内存缓存区中,直接调用1的getBitmap()函数,获取bitmap数据(先在硬引用缓存区查找匹配,若硬引用区匹配失败,再去软引用区匹配)

2) 如果从内存缓存区读取失败,再从外部文件缓存中读取,调用2的getBitmap()函数

3) 如果从外部文件缓存中读取失败,则从服务端下载该图片,过程3.

5.生成key值

[html] view plaincopy

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值

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 221,820评论 6 515
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,648评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 168,324评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,714评论 1 297
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,724评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,328评论 1 310
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,897评论 3 421
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,804评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,345评论 1 318
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,431评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,561评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,238评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,928评论 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,417评论 0 24
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,528评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,983评论 3 376
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,573评论 2 359

推荐阅读更多精彩内容