相册图片预取缓存策略是内存缓存(硬引用LruCache、软引用SoftReference<Bitmap>)、外部文件缓存(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<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.外部文件缓存
[html] view plaincopy
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.从服务端下载图片
 
下载成功后调用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值

Android 缓存机制相关推荐

  1. Android 缓存机制 LRUCache

    1.缓存策略 缓存策略主要包含缓存的添加.获取和删除三类操作.删除缓存是因为不管是内存缓存还是硬盘缓存,它们的缓存大小都是有限的.当缓存满了之后,再想添加缓存就需要删除一些旧的缓存来添加新的缓存. L ...

  2. android lru进程队列,Android缓存机制——LruCache

    一.Android中的缓存策略 Android的缓存,主要的就是内存缓存和硬盘缓存. 不管是内存缓存还是硬盘缓存,它们的缓存大小都是有限的.当缓存满了之后,再想其添加缓存,这个时候就需要删除一些旧的缓 ...

  3. Android 缓存

    1.Android缓存机制&一个缓存框架推荐 http://blog.csdn.net/shakespeare001/article/details/51695358 2.ASimpleCac ...

  4. android 视频的缩略图 缓存机制和 异步加载缩略图

    在这次的工作开发项目中,涉及到一个视频缩略图的视频列表:这个在大家看来,制作视频缩略图就是两行代码就搞定的事.确实是这样的,百度一下,每个帖子都知道制作视频缩略图的方法,在这里确实也是一样的,但是我要 ...

  5. 【腾讯Bugly干货分享】Android ListView与RecyclerView对比浅析--缓存机制

    本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/5811d... 作者:黄宁源 一,背景 RecyclerView是谷歌官方出的一 ...

  6. Android笔记(二十五) ListView的缓存机制与BaseAdapter

    之前接触了ListView和Adapter,Adapter将数据源和View连接起来,实际应用中,我们要显示的数据往往有很多,而屏幕只有那么大,系统只能屏幕所能显示的内容,当我们滑动屏幕,会将旧的内容 ...

  7. Android学习——ListView的缓存机制

    在使用ListView的时候,需要加载适配器和数据源,这篇文章主要介绍一下ListView的使用以及利用ListView的缓存机制来减少系统的初始化时间. ListView的使用 ListView和V ...

  8. android webView的缓存机制和资源预加载

    android 原生使用WebView嵌入H5页面 Hybrid开发 一.性能问题 android webview 里H5加载速度慢 网络流量大 1.H5页面加载速度慢 渲染速度慢 js解析效率 js ...

  9. Android Glide图片加载框架(三)缓存机制

    文章目录 一.缓存简介 二.缓存用法 内存缓存方式 磁盘缓存方式 三.缓存KEY 四.内存缓存 内存缓存流程 五.磁盘缓存 磁盘缓存流程 Android Glide图片加载框架系列文章 Android ...

最新文章

  1. [CF482B]Interesting Array
  2. Merge into的使用详解-你Merge了没有
  3. JAVA WEB部分易混淆问题总结
  4. linux日志文件存放目录,Log4j 日志文件Linux/Mac/Windows通用存放位置设置方法
  5. exif linux php扩展_LNMP环境为PHP添加exif扩展
  6. mysql 修改字段长度_面试官:InnoDB记录存储结构都不知道,你敢说你懂MySQL?
  7. sql创建表主键gui_在SQL Server中使用主数据服务快速创建最终用户可以维护的GUI
  8. linux telnet 安装
  9. C风格简易本地log系统
  10. Give root password for maintenance 问题的解决方法
  11. 四川绵阳:充分利用区块链等技术,为农民工证照办理提供线上便捷服务
  12. excel小技巧1:修改的日期格式为什么要双击一下单元格才能变
  13. python批量生成列表_Python编程实战:根据数据表格,快速批量生成符合规范的WORD文档...
  14. 想当然很可怕,根子还是思考上的懒惰
  15. 编译UE4出现C1076错误的解决方法
  16. 又一大型色情直播App被捣毁,女主播哭求别告诉家人
  17. 《西西弗神话》笔记——不期待人生可以过得很顺利,但我希望碰到人生难关的时候,自己可以是它的对手
  18. 2015年百度之星程序设计大赛 - 初赛(2)
  19. 使用高级语言编写计算机程序步骤,计算机执行用高级语言编写的程序主要有两种途径解释和编译编译专.doc...
  20. c语言 百度文库,百度文库C语言专本辅导第一二章.doc

热门文章

  1. python函数调用位置_Python: 浅谈函数局部变量快在哪
  2. Codeforces 987B. High School: Become Human
  3. x86服务器当虚拟化的存储,X86服务器虚拟化实施方案.doc
  4. mysql慢查询单位_MySQL慢查询
  5. 1024 科学计数法 (20 分)(c语言)
  6. Navicat导入TXT到数据库
  7. C语言面向对象编程(六):配置文件解析
  8. linux服务器cuda,cudnn的安装与卸载
  9. 巨坑 之 pip install 和 conda install 的区别 以及 查看 和 修改 虚拟环境下运行路径
  10. c++ primer 5th,练习11.19,编写代码验证