最近开发电视版的云存储应用,要求”我的相册“模块有全屏预览图片的功能,全屏分辨率是1920*1080超清。
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读取大图片并缓存的更多相关文章

  1. android对大图片的缓存处理

    废话不多说,直接上代码 package com.huge.emj.common.util; import java.io.File; import java.io.FileInputStream; i ...

  2. Android异步下载图片并且缓存图片到本地

    Android异步下载图片并且缓存图片到本地 在Android开发中我们经常有这样的需求,从服务器上下载xml或者JSON类型的数据,其中包括一些图片资源,本demo模拟了这个需求,从网络上加载XML ...

  3. android读取远程图片案例

    关键代码:Bitmap bitmap=BitmapFactory.decodeByteArray(data, 0, data.length);imageview.setImageBitmap(bitm ...

  4. Android读取url图片保存及文件读取

    参考: 1.http://blog.csdn.net/ameyume/article/details/6528205 2.http://blog.sina.com.cn/s/blog_85b3a161 ...

  5. Android 显示大图片

    主要的代码如下: BitmapFactory.Options options = new BitmapFactory.Options(); //图片解析配置 options.inJustDecodeB ...

  6. android 应用开发对大图片的处理

    一,下载 android下载大图片(例如微博长图片)会出现OOM down掉问题 解决这个问题的办法是下载图片时先得到图片的宽度和高度,如果超出规定限制则对图片进行缩放 关键参数 1. BitmapF ...

  7. Android调用系统相机、自己定义相机、处理大图片

    Android调用系统相机和自己定义相机实例 本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,而且因为涉及到要把拍到的照片显示出来,该样例也会涉及到Android载入大图片时候的处 ...

  8. Android Camera开发系列(上)——Camera的基本调用与实现拍照功能以及获取拍照图片加载大图片

    Android Camera开发系列(上)--Camera的基本调用与实现拍照功能以及获取拍照图片加载大图片 最近也是在搞个破相机,兼容性那叫一个不忍直视啊,于是自己翻阅了一些基本的资料,自己实现了一 ...

  9. java快速获取大图片的分辨率(大图片格式JPG,tiff ,eg)

    问题描述:怎样快速获取一个20MB图片的分辨率? 程序代码: package test; import java.awt.Dimension; import java.awt.image.Buffer ...

随机推荐

  1. Atitit 图像处理之编程之类库调用的接口api cli gui ws rest  attilax大总结.docx

    Atitit 图像处理之编程之类库调用的接口api cli gui ws rest  attilax大总结.docx 1. 为什么需要接口调用??1 1.1. 为了方便集成复用模块类库1 1.2. 嫁 ...

  2. 互斥锁pthread_mutex_init()函数

    linux下为了多线程同步,通常用到锁的概念.posix下抽象了一个锁类型的结构:ptread_mutex_t.通过对该结构的操作,来判断资源是否可以访问.顾名思义,加锁(lock)后,别人就无法打开 ...

  3. sql中计算某天是全年的第几周及取得某天的所在周的周一的日期的函数

    --取得某天的所在周的周一的函数 CREATE FUNCTION getMondayBtDate(@date datetime) RETURNS date AS begin DECLARE @week ...

  4. JAVA与C++,C与C++的差别

    首先来分析JAVA与C++的差别: JAVA是纯面向对象的语言,而C++是基于面向对象过程的语言. JAVA有着垃圾回收机制.它的语法是C++的子集,即JAVA有的C++都有.而C++有的JAVA不全 ...

  5. nfs 客户端启动报错rpc.mountd: svc_tli_create: could not open connection for tcp6

    # /etc/init.d/nfs start Starting NFS services: [ OK ] Starting NFS mountd: rpc.mountd: svc_tli_creat ...

  6. schema中字段类型的定义

    当schema中字段类型为String时,保存的时候如果该字段为Number也可以保存成功,mongoose会自动将其转换为数字字符串. 当schema中字段类型为Number时,保存的时候如果该字段 ...

  7. HBase最佳实践-用好你的操作系统

    终于又切回HBase模式了,之前一段时间因为工作的原因了解接触了一段时间大数据生态的很多其他组件(诸如Parquet.Carbondata.Hive.SparkSQL.TPC-DS/TPC-H等),虽 ...

  8. 【Linux技术】BusyBox详解

    BusyBox 是很多标准 Linux® 工具的一个单个可执行实现.BusyBox 包含了一些简单的工具,例如 cat 和 echo,还包含了一些更大.更复杂的工具,例如 grep.find.moun ...

  9. 【Socket】linux下http服务器开发

    1.mystery引入 1)超文本传输协议(HTTP)是一种应用于分布式.合作式.多媒体信息系统的应用层协议 2)工作原理 1)客户端一台客户机与服务器建立连接后,会发送一个请求给服务器,请求方式的格 ...

  10. ubuntu创建用户命令

    可以用adduser和useradd来添加用户,用userdel来删除用户. 最简单的命令:sudo adduser test会自动同名组,创建/home/test/,从etc/skel/复制文件,并 ...