`
寻梦者
  • 浏览: 638134 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

android读取大图片并缓存

 
阅读更多

 

最近开发电视版的云存储应用,要求”我的相册“模块有全屏预览图片的功能,全屏分辨率是1920*1080超清。

UI组件方面采用Gallery+ImageSwitcher组合,这里略过,详情参见google Android API。

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

1.内存缓存

 

[html] view plaincopy
  1. //需要导入外部jar文件 android-support-v4.jar  
  2.     import android.support.v4.util.LruCache;  
  3.     //开辟8M硬缓存空间  
  4.     private final int hardCachedSize = 8*1024*1024;       
  5.     //hard cache  
  6.     private final LruCache<String, Bitmap> sHardBitmapCache = new LruCache<String, Bitmap>(hardCachedSize){  
  7.         @Override  
  8.         public int sizeOf(String key, Bitmap value){  
  9.             return value.getRowBytes() * value.getHeight();  
  10.         }  
  11.         @Override  
  12.         protected void entryRemoved(boolean evicted, String key, Bitmap oldValue, Bitmap newValue){  
  13.             Log.v("tag", "hard cache is full , push to soft cache");  
  14.             //硬引用缓存区满,将一个最不经常使用的oldvalue推入到软引用缓存区  
  15.             sSoftBitmapCahe.put(key, new SoftReference<Bitmap>(oldValue));  
  16.         }  
  17.     }  
  18.     //软引用  
  19.     private static final int SOFT_CACHE_CAPACITY = 40;  
  20.     private final static LinkedHashMap<String, SoftReference<Bitmap>> sSoftBitmapCache =   
  21.         new  LinkedHashMao<String, SoftReference<Bitmap>>(SOFT_CACHE_CAPACITY, 0.75f, true){  
  22.         @Override  
  23.         public SoftReference<Bitmap> put(String key, SoftReference<Bitmap> value){  
  24.             return super.input(key, value);  
  25.         }  
  26.         @Override  
  27.         protected boolean removeEldestEntry(LinkedHashMap.Entry<Stirng, SoftReference<Bitmap>> eldest){  
  28.             if(size() > SOFT_CACHE_CAPACITY){  
  29.                 Log.v("tag", "Soft Reference limit , purge one");  
  30.                 return true;  
  31.             }  
  32.             return false;  
  33.         }  
  34.     }  
  35.     //缓存bitmap  
  36.     public boolean putBitmap(String key, Bitmap bitmap){  
  37.         if(bitmap != null){  
  38.             synchronized(sHardBitmapCache){  
  39.                 sHardBitmapCache.put(key, bitmap);  
  40.             }  
  41.             return true;  
  42.         }         
  43.         return false;  
  44.     }  
  45.     //从缓存中获取bitmap  
  46.     public Bitmap getBitmap(String key){  
  47.         synchronized(sHardBitmapCache){  
  48.             final Bitmap bitmap = sHardBitmapCache.get(key);  
  49.             if(bitmap != null)  
  50.                 return bitmap;  
  51.         }  
  52.         //硬引用缓存区间中读取失败,从软引用缓存区间读取  
  53.         synchronized(sSoftBitmapCache){  
  54.             SoftReference<Bitmap> bitmapReference = sSoftBtimapCache.get(key);  
  55.             if(bitmapReference != null){  
  56.                 final Bitmap bitmap2 = bitmapReference.get();  
  57.                 if(bitmap2 != null)  
  58.                     return bitmap2;  
  59.                 else{  
  60.                     Log.v("tag", "soft reference 已经被回收");  
  61.                     sSoftBitmapCache.remove(key);  
  62.                 }  
  63.             }  
  64.         }  
  65.         return null;  
  66.     }  

 

2.外部文件缓存

[html] view plaincopy
  1. private File mCacheDir = context.getCacheDir();  
  2.     private static final int MAX_CACHE_SIZE = 20 * 1024 * 1024; //20M  
  3.     private final LruCache<String, Long> sFileCache = new LruCache<String, Long>(MAX_CACHE_SIZE){  
  4.         @Override  
  5.         public int sizeOf(String key, Long value){  
  6.             return value.intValue();  
  7.         }  
  8.         @Override  
  9.         protected void entryRemoved(boolean evicted, String key, Long oldValue, Long newValue){  
  10.             File file = getFile(key);  
  11.             if(file != null)  
  12.                 file.delete();  
  13.         }  
  14.     }  
  15.     private File getFile(String fileName) throws FileNotFoundException {  
  16.         File file = new File(mCacheDir, fileName);  
  17.         if(!file.exists() || !file.isFile())  
  18.             throw new FileNotFoundException("文件不存在或有同名文件夹");  
  19.         return file;  
  20.     }  
  21.     //缓存bitmap到外部存储  
  22.     public boolean putBitmap(String key, Bitmap bitmap){  
  23.         File file = getFile(key);  
  24.         if(file != null){  
  25.             Log.v("tag", "文件已经存在");  
  26.             return true;  
  27.         }  
  28.         FileOutputStream fos = getOutputStream(key);  
  29.         boolean saved = bitmap.compress(CompressFormat.JPEG, 100, fos);  
  30.         fos.flush();  
  31.         fos.close();  
  32.         if(saved){  
  33.             synchronized(sFileCache){  
  34.                 sFileCache.put(key, getFile(key).length());  
  35.             }  
  36.             return true;   
  37.         }  
  38.         return false;  
  39.     }  
  40.     //根据key获取OutputStream  
  41.     private FileOutputStream getOutputStream(String key){  
  42.         if(mCacheDir == null)  
  43.             return null;  
  44.         FileOutputStream fos = new FileOutputStream(mCacheDir.getAbsolutePath() + File.separator + key);  
  45.         return fos;  
  46.     }  
  47.     //获取bitmap  
  48.     private static BitmapFactory.Options sBitmapOptions;  
  49.     static {  
  50.         sBitmapOptions = new BitmapFactory.Options();  
  51.         sBitmapOptions.inPurgeable=true; //bitmap can be purged to disk  
  52.     }  
  53.     public Bitmap getBitmap(String key){  
  54.         File bitmapFile = getFile(key);  
  55.         if(bitmapFile != null){  
  56.             Bitmap bitmap = BitmapFactory.decodeStream(new FileInputStream(bitmapFile), null, sBitmapOptions);  
  57.             if(bitmap != null){  
  58.                 //重新将其缓存至硬引用中  
  59.                 ...  
  60.             }  
  61.         }  
  62.     }  

 

3.从服务端下载图片

 

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

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

 

4.预览图片的流程

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

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

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

 

5.生成key值

[html] view plaincopy
  1. private static String generateKey(String fileId, int width, int height) {         
  2.         String ret = fileId + "_" + Integer.toString(width) + "x" + Integer.toString(height);  
  3.         return ret;  
  4.     }  
  5.     String key = generateKey(...)即可生成唯一的key值  
分享到:
评论
5 楼 Kinphoo 2014-01-04  
,感谢分享~~
4 楼 Cindy_Lee 2013-10-29  
你这么做可能有问题啊 没有考虑bitmap的生命周期。如果某个view被回收了,里面用到的bitmap系统会自动recycle,有因为这个bitmap的引用在你的lrucache里面也有,这样获取的这个bitmap的状态可能就是recycled的,这样就能会报错。
3 楼 belmen 2013-02-04  
Android系统中SoftReference没什么用的,放进去的等你读出来基本都被销毁掉了,可以去掉
2 楼 mikelvin 2012-05-08  
小错误不少
1 楼 keeponmoving 2012-05-03  
M一下,正是我要看的,感谢楼主。

相关推荐

    android 下载图片并缓存

    - **失效时间**:设置缓存图片的过期时间,超过这个时间则从服务器重新下载。 - **强制缓存**:在网络不稳定或无网络时,优先从本地缓存读取图片。 - **条件请求**:使用If-None-Match或If-Modified-Since头进行...

    Android图片缓存框架Fresco极其强大的配置

    Fresco通过其独特的架构,避免了OOM(Out of Memory)错误,并提供了高效的图片缓存策略。本文将深入探讨Fresco的配置选项及其工作原理。 ### 一、Fresco的架构 Fresco的核心架构分为三个层次:**内存缓存**、**...

    Android例子源码异步批量下载图片并缓存

    在Android开发中,异步批量下载图片并缓存是一个常见的需求,特别是在开发涉及大量图片展示的应用时,如社交应用、电商应用等。本教程将基于提供的Android例子源码,深入探讨如何实现这一功能。 首先,我们需要理解...

    volley获取图片并缓存

    在Android应用中,加载图片是一项常见的任务,而Volley提供了强大的图片加载和缓存功能,使得即使在离线状态下,用户也能查看之前加载过的图片。 Volley的图片缓存机制主要包括两个部分:内存缓存和磁盘缓存。内存...

    Android批量下载图片并缓存,非常流畅

    本教程将详细讲解如何在Android应用中实现批量下载图片并进行高效缓存,以实现非常流畅的用户体验。我们将主要关注LruCache技术,这是一种内存管理策略,有助于优化内存使用。 首先,我们需要理解Android中的图片...

    Android获取手机所有图片并显示

    在Android开发中,获取手机上的所有图片并进行列表展示是一项常见的需求,这通常涉及到多媒体文件的访问、数据的加载和UI的设计。以下是一些相关的知识点: 1. **多媒体存储权限**: 在Android 6.0(API级别23)及...

    android获取本地图片相册及图片

    在Android开发中,获取本地图片相册以及展示图片是一项常见的任务。这涉及到用户界面的设计、权限管理、多媒体数据的读取以及图片加载等多个知识点。以下将详细解释这些关键点。 首先,我们需要理解Android的文件...

    android 图片的二级缓存

    当应用需要加载图片时,首先会尝试从内存缓存中查找,如果找不到,则从磁盘缓存中读取,如果磁盘缓存也没有,再从网络或本地文件系统加载,并将图片存入两级缓存中,以便后续使用。 二、内存缓存 内存缓存通常使用...

    Android远程图片获取和本地缓存

    在Android应用开发中,远程图片的获取和本地缓存是一个重要的优化策略,旨在提升用户体验,避免因长时间等待网络加载而导致的无响应异常。通常,Android系统会在UI线程执行耗时操作超过5秒时抛出ANR(Application ...

    安卓源码包android图片缓存&展示Android 异步加载图片等24个合集.zip

    安卓源码包android图片缓存&展示Android 异步加载图片等24个合集: ‘360全景查看demo.rar afinal框架实现图片的简单异步缓存加载.rar andengine中直接加载多张小图片合成一张大图片生成动画精灵.rar android gif...

    Android获取第三方应用的占用大小,包括缓存、数据、应用大小

    Android通过IPackageStatsObserver.aidl 、PackageStats.aidl两个AIDL文件获取第三方应用的占用大小,包括缓存、数据、应用大小,经验证和手机设置里面显示的大小完全一致。。。。。。如果还需要其他比如清除缓存之...

    android 网络图片双缓存

    所以设计了一个图片缓存技术来解决每次android手机加载图片的问题 内存的读取速度是最快的,然后是文件的读取速度,最后是网络资源的读取 既然内存的读取时间最快,我们好好利用内存资源。将内存再分两层...

    Android大图片缓存技术Demo

    本项目“Android大图片缓存技术Demo”旨在演示如何有效解决这个问题,通过运用图片压缩技术和缓存策略,确保应用能够流畅地加载大量图片而不会引发性能问题。 首先,我们要理解Android中的内存管理机制。Android...

    Android中Glide获取缓存大小并清除缓存图片

    Android中Glide获取缓存大小并清除缓存图片是 Android 开发中一个重要的知识点,了解 Glide 获取缓存大小并清除缓存图片可以帮助开发者更好地管理应用程序的缓存,避免缓存过大占用太多的存储空间。下面将详细介绍 ...

    Android Google官网的图片缓存源码

    本文将深入探讨Android图片缓存的基本原理,并结合Google的实现方式,解析其中的关键知识点。 首先,图片缓存通常包括两大部分:内存缓存(Memory Cache)和磁盘缓存(Disk Cache)。内存缓存是将图片数据存储在...

    Android 图片缓存

    在Android开发中,图片缓存是一项重要的技术,用于优化应用性能和减少网络资源的消耗。在用户滚动列表或浏览大量图片时,如果没有合适的缓存机制,可能会导致内存溢出(OOM)或者用户体验下降,因为频繁的网络请求会...

    Android 图片三级缓存的一个好用demo(ImageLoaderDemo)

    "Android 图片三级缓存的一个好用demo(ImageLoaderDemo)" 提供了一个有效的解决方案,旨在提高图片加载速度并降低对网络资源的依赖。这个Demo主要涉及了Android中的图片缓存策略,包括内存缓存、磁盘缓存以及网络...

    Android大量图片加载的3级缓存

    "Android大量图片加载的3级缓存"是一个关键的概念,它旨在优化用户体验,减少内存消耗,并提高应用程序性能。以下是这个主题的详细解释。 ### 1. 图片压缩 在Android中,图片加载前进行压缩是非常重要的,因为未经...

    Android图片缓存

    在Android应用开发中,图片加载是一项常见的任务,但频繁地从网络下载或读取本地大图可能会消耗大量系统资源,导致应用性能下降甚至出现内存溢出等问题。为了解决这些问题,开发者通常会采用图片缓存机制。本篇将...

    Android图片的压缩与缓存

    本篇将深入探讨Android图片的压缩与缓存机制,以提高应用性能并防止内存溢出。 ### Bitmap对象与内存管理 Bitmap是Android中用于表示图像数据的类,它直接占用大量的内存资源。当加载大尺寸图片时,未进行适当处理...

Global site tag (gtag.js) - Google Analytics