转 如何高效使用和管理Bitmap--图片缓存管理模块的设计与实现
上周为360全景项目引入了图片缓存模块。因为是在Android4.0平台以上运作,出于惯性,都会在设计之前查阅相关资料,尽量避免拿一些以前2.3平台积累的经验来进行类比处理。开发文档中有一个 BitmapFun的示例,仔细拜读了一下,虽说围绕着Bitmap的方方面面讲得都很深入,但感觉很难引入到当前项目中去。 现在的图片服务提供者基本上都来源于网络。对于应用平台而言,访问网络属于耗时操作。尤其是在移动终端设备上,它的显著表现为系统的延迟时间变长、用户交互性变差等。可以想象,一个携带着这些问题的应用在市场上是很难与同类产品竞争的。
说明一下,本文借鉴了 Keegan小钢和安卓巴士的处理模板,主要针对的是4.0以上平台应用。2.3以前平台执行效果未知,请斟酌使用或直接略过:),当然更欢迎您把测试结果告知笔者。
一、图片加载流程
首先,我们谈谈加载图片的流程,项目中的该模块处理流程如下:
1.在UI主线程中,从内存缓存中获取图片,找到后返回。找不到进入下一步;
2.在工作线程中,从磁盘缓存中获取图片,找到即返回并更新内存缓存。找不到进入下一步;
3.在工作线程中,从网络中获取图片,找到即返回并同时更新内存缓存和磁盘缓存。找不到显示默认以提示。
二、内存缓存类(PanoMemCache)
这里使用Android提供的LruCache类,该类保存一个强引用来限制内容数量,每当Item被访问的时候,此Item就会移动到队列的头部。当cache已满的时候加入新的item时,在队列尾部的item会被回收。
- public class PanoMemoryCache {
- // LinkedHashMap初始容量
- private static final int INITIAL_CAPACITY = 16;
- // LinkedHashMap加载因子
- private static final int LOAD_FACTOR = 0.75f;
- // LinkedHashMap排序模式
- private static final boolean ACCESS_ORDER = true;
- // 软引用缓存
- private static LinkedHashMap<String, SoftReference<Bitmap>> mSoftCache;
- // 硬引用缓存
- private static LruCache<String, Bitmap> mLruCache;
- public PanoMemoryCache() {
- // 获取单个进程可用内存的最大值
- // 方式一:使用ActivityManager服务(计量单位为M)
- /*int memClass = ((ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE)).getMemoryClass();*/
- // 方式二:使用Runtime类(计量单位为Byte)
- final int memClass = (int) Runtime.getRuntime().maxMemory();
- // 设置为可用内存的1/4(按Byte计算)
- final int cacheSize = memClass / 4;
- mLruCache = new LruCache<String, Bitmap>(cacheSize) {
- @Override
- protected int sizeOf(String key, Bitmap value) {
- if(value != null) {
- // 计算存储bitmap所占用的字节数
- return value.getRowBytes() * value.getHeight();
- } else {
- return 0;
- }
- }
- @Override
- protected void entryRemoved(boolean evicted, String key, Bitmap oldValue, Bitmap newValue) {
- if(oldValue != null) {
- // 当硬引用缓存容量已满时,会使用LRU算法将最近没有被使用的图片转入软引用缓存
- mSoftCache.put(key, new SoftReference<Bitmap>(oldValue));
- }
- }
- };
- /*
- * 第一个参数:初始容量(默认16)
- * 第二个参数:加载因子(默认0.75)
- * 第三个参数:排序模式(true:按访问次数排序;false:按插入顺序排序)
- */
- mSoftCache = new LinkedHashMap<String, SoftReference<Bitmap>>(INITIAL_CAPACITY, LOAD_FACTOR, ACCESS_ORDER) {
- private static final long serialVersionUID = 7237325113220820312L;
- @Override
- protected boolean removeEldestEntry(Entry<String, SoftReference<Bitmap>> eldest) {
- if(size() > SOFT_CACHE_SIZE) {
- return true;
- }
- return false;
- }
- };
- }
- /**
- * 从缓存中获取Bitmap
- * @param url
- * @return bitmap
- */
- public Bitmap getBitmapFromMem(String url) {
- Bitmap bitmap = null;
- // 先从硬引用缓存中获取
- synchronized (mLruCache) {
- bitmap = mLruCache.get(url);
- if(bitmap != null) {
- // 找到该Bitmap之后,将其移到LinkedHashMap的最前面,保证它在LRU算法中将被最后删除。
- mLruCache.remove(url);
- mLruCache.put(url, bitmap);
- return bitmap;
- }
- }
- // 再从软引用缓存中获取
- synchronized (mSoftCache) {
- SoftReference<Bitmap> bitmapReference = mSoftCache.get(url);
- if(bitmapReference != null) {
- bitmap = bitmapReference.get();
- if(bitmap != null) {
- // 找到该Bitmap之后,将它移到硬引用缓存。并从软引用缓存中删除。
- mLruCache.put(url, bitmap);
- mSoftCache.remove(url);
- return bitmap;
- } else {
- mSoftCache.remove(url);
- }
- }
- }
- return null;
- }
- /**
- * 添加Bitmap到内存缓存
- * @param url
- * @param bitmap
- */
- public void addBitmapToCache(String url, Bitmap bitmap) {
- if(bitmap != null) {
- synchronized (mLruCache) {
- mLruCache.put(url, bitmap);
- }
- }
- }
- /**
- * 清理软引用缓存
- */
- public void clearCache() {
- mSoftCache.clear();
- mSoftCache = null;
- }
- }
补充一点,由于4.0平台以后对SoftReference类引用的对象调整了回收策略,所以该类中的软引用缓存实际上没什么效果,可以去掉。2.3以前平台建议保留。
三、磁盘缓存类(PanoDiskCache)
- public class PanoDiskCache {
- private static final String TAG = "PanoDiskCache";
- // 文件缓存目录
- private static final String CACHE_DIR = "panoCache";
- private static final String CACHE_FILE_SUFFIX = ".cache";
- private static final int MB = 1024 * 1024;
- private static final int CACHE_SIZE = 10; // 10M
- private static final int SDCARD_CACHE_THRESHOLD = 10;
- public PanoDiskCache() {
- // 清理文件缓存
- removeCache(getDiskCacheDir());
- }
- /**
- * 从磁盘缓存中获取Bitmap
- * @param url
- * @return
- */
- public Bitmap getBitmapFromDisk(String url) {
- String path = getDiskCacheDir() + File.separator + genCacheFileName(url);
- File file = new File(path);
- if(file.exists()) {
- Bitmap bitmap = BitmapFactory.decodeFile(path);
- if(bitmap == null) {
- file.delete();
- } else {
- updateLastModified(path);
- return bitmap;
- }
- }
- return null;
- }
- /**
- * 将Bitmap写入文件缓存
- * @param bitmap
- * @param url
- */
- public void addBitmapToCache(Bitmap bitmap, String url) {
- if(bitmap == null) {
- return;
- }
- // 判断当前SDCard上的剩余空间是否足够用于文件缓存
- if(SDCARD_CACHE_THRESHOLD > calculateFreeSpaceOnSd()) {
- return;
- }
- String fileName = genCacheFileName(url);
- String dir = getDiskCacheDir();
- File dirFile = new File(dir);
- if(!dirFile.exists()) {
- dirFile.mkdirs();
- }
- File file = new File(dir + File.separator + fileName);
- try {
- file.createNewFile();
- FileOutputStream out = new FileOutputStream(file);
- bitmap.compress(Bitmap.CompressFormat.JPEG, 100, out);
- out.flush();
- out.close();
- } catch (FileNotFoundException e) {
- Log.e(TAG, "FileNotFoundException");
- } catch (IOException e) {
- Log.e(TAG, "IOException");
- }
- }
- /**
- * 清理文件缓存
- * 当缓存文件总容量超过CACHE_SIZE或SDCard的剩余空间小于SDCARD_CACHE_THRESHOLD时,将删除40%最近没有被使用的文件
- * @param dirPath
- * @return
- */
- private boolean removeCache(String dirPath) {
- File dir = new File(dirPath);
- File[] files = dir.listFiles();
- if(files == null || files.length == 0) {
- return true;
- }
- if(!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
- return false;
- }
- int dirSize = 0;
- for (int i = 0; i < files.length; i++) {
- if(files[i].getName().contains(CACHE_FILE_SUFFIX)) {
- dirSize += files[i].length();
- }
- }
- if(dirSize > CACHE_SIZE * MB || SDCARD_CACHE_THRESHOLD > calculateFreeSpaceOnSd()) {
- int removeFactor = (int) (0.4 * files.length + 1);
- Arrays.sort(files, new FileLastModifiedSort());
- for (int i = 0; i < removeFactor; i++) {
- if(files[i].getName().contains(CACHE_FILE_SUFFIX)) {
- files[i].delete();
- }
- }
- }
- if(calculateFreeSpaceOnSd() <= SDCARD_CACHE_THRESHOLD) {
- return false;
- }
- return true;
- }
- /**
- * 更新文件的最后修改时间
- * @param path
- */
- private void updateLastModified(String path) {
- File file = new File(path);
- long time = System.currentTimeMillis();
- file.setLastModified(time);
- }
- /**
- * 计算SDCard上的剩余空间
- * @return
- */
- private int calculateFreeSpaceOnSd() {
- StatFs stat = new StatFs(Environment.getExternalStorageDirectory().getPath());
- double sdFreeMB = ((double) stat.getAvailableBlocks() * (double) stat.getBlockSize()) / MB;
- return (int) sdFreeMB;
- }
- /**
- * 生成统一的磁盘文件后缀便于维护
- * 从URL中得到源文件名称,并为它追加缓存后缀名.cache
- * @param url
- * @return 文件存储后的名称
- */
- private String genCacheFileName(String url) {
- String[] strs = url.split(File.separator);
- return strs[strs.length - 1] + CACHE_FILE_SUFFIX;
- }
- /**
- * 获取磁盘缓存目录
- * @return
- */
- private String getDiskCacheDir() {
- return getSDPath() + File.separator + CACHE_DIR;
- }
- /**
- * 获取SDCard目录
- * @return
- */
- private String getSDPath() {
- File sdDir = null;
- // 判断SDCard是否存在
- boolean sdCardExist = Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED);
- if(sdCardExist) {
- // 获取SDCard根目录
- sdDir = Environment.getExternalStorageDirectory();
- }
- if(sdDir != null) {
- return sdDir.toString();
- } else {
- return "";
- }
- }
- /**
- * 根据文件最后修改时间进行排序
- */
- private class FileLastModifiedSort implements Comparator<File> {
- @Override
- public int compare(File lhs, File rhs) {
- if(lhs.lastModified() > rhs.lastModified()) {
- return 1;
- } else if(lhs.lastModified() == rhs.lastModified()) {
- return 0;
- } else {
- return -1;
- }
- }
- }
- }
四、图片工具类(PanoUtils)
1.从网络上获取图片:downloadBitmap()
- /**
- * 从网络上获取Bitmap,并进行适屏和分辨率处理。
- * @param context
- * @param url
- * @return
- */
- public static Bitmap downloadBitmap(Context context, String url) {
- HttpClient client = new DefaultHttpClient();
- HttpGet request = new HttpGet(url);
- try {
- HttpResponse response = client.execute(request);
- int statusCode = response.getStatusLine().getStatusCode();
- if(statusCode != HttpStatus.SC_OK) {
- Log.e(TAG, "Error " + statusCode + " while retrieving bitmap from " + url);
- return null;
- }
- HttpEntity entity = response.getEntity();
- if(entity != null) {
- InputStream in = null;
- try {
- in = entity.getContent();
- return scaleBitmap(context, readInputStream(in));
- } finally {
- if(in != null) {
- in.close();
- in = null;
- }
- entity.consumeContent();
- }
- }
- } catch (IOException e) {
- request.abort();
- Log.e(TAG, "I/O error while retrieving bitmap from " + url, e);
- } catch (IllegalStateException e) {
- request.abort();
- Log.e(TAG, "Incorrect URL: " + url);
- } catch (Exception e) {
- request.abort();
- Log.e(TAG, "Error while retrieving bitmap from " + url, e);
- } finally {
- client.getConnectionManager().shutdown();
- }
- return null;
- }
2.从输入流读取字节数组,看起来是不是很眼熟啊!
- public static byte[] readInputStream(InputStream in) throws Exception {
- ByteArrayOutputStream out = new ByteArrayOutputStream();
- byte[] buffer = new byte[1024];
- int len = 0;
- while((len = in.read(buffer)) != -1) {
- out.write(buffer, 0, len);
- }
- in.close();
- return out.toByteArray();
- }
3.对下载的源图片进行适屏处理,这也是必须的:)
- /**
- * 按使用设备屏幕和纹理尺寸适配Bitmap
- * @param context
- * @param in
- * @return
- */
- private static Bitmap scaleBitmap(Context context, byte[] data) {
- WindowManager windowMgr = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
- DisplayMetrics outMetrics = new DisplayMetrics();
- windowMgr.getDefaultDisplay().getMetrics(outMetrics);
- int scrWidth = outMetrics.widthPixels;
- int scrHeight = outMetrics.heightPixels;
- BitmapFactory.Options options = new BitmapFactory.Options();
- options.inJustDecodeBounds = true;
- Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length, options);
- int imgWidth = options.outWidth;
- int imgHeight = options.outHeight;
- if(imgWidth > scrWidth || imgHeight > scrHeight) {
- options.inSampleSize = calculateInSampleSize(options, scrWidth, scrHeight);
- }
- options.inJustDecodeBounds = false;
- bitmap = BitmapFactory.decodeByteArray(data, 0, data.length, options);
- // 根据业务的需要,在此处还可以进一步做处理
- ...
- return bitmap;
- }
- /**
- * 计算Bitmap抽样倍数
- * @param options
- * @param reqWidth
- * @param reqHeight
- * @return
- */
- public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
- // 原始图片宽高
- final int height = options.outHeight;
- final int width = options.outWidth;
- int inSampleSize = 1;
- if (height > reqHeight || width > reqWidth) {
- // 计算目标宽高与原始宽高的比值
- final int heightRatio = Math.round((float) height / (float) reqHeight);
- final int widthRatio = Math.round((float) width / (float) reqWidth);
- // 选择两个比值中较小的作为inSampleSize的值
- inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;
- if(inSampleSize < 1) {
- inSampleSize = 1;
- }
- }
- return inSampleSize;
- }
五、使用decodeByteArray()还是decodeStream()?
讲到这里,有童鞋可能会问我为什么使用BitmapFactory.decodeByteArray(data, 0, data.length, opts)来创建Bitmap,而非使用BitmapFactory.decodeStream(is, null, opts)。你这样做不是要多写一个静态方法readInputStream()吗?
没错,decodeStream()确实是该使用情景下的首选方法,但是在有些情形下,它会导致图片资源不能即时获取,或者说图片被它偷偷地缓存起来,交 还给我们的时间有点长。但是延迟性是致命的,我们等不起。所以在这里选用decodeByteArray()获取,它直接从字节数组中获取,贴近于底层 IO、脱离平台限制、使用起来风险更小。
六、引入缓存机制后获取图片的方法
- /**
- * 加载Bitmap
- * @param url
- * @return
- */
- private Bitmap loadBitmap(String url) {
- // 从内存缓存中获取,推荐在主UI线程中进行
- Bitmap bitmap = memCache.getBitmapFromMem(url);
- if(bitmap == null) {
- // 从文件缓存中获取,推荐在工作线程中进行
- bitmap = diskCache.getBitmapFromDisk(url);
- if(bitmap == null) {
- // 从网络上获取,不用推荐了吧,地球人都知道~_~
- bitmap = PanoUtils.downloadBitmap(this, url);
- if(bitmap != null) {
- diskCache.addBitmapToCache(bitmap, url);
- memCache.addBitmapToCache(url, bitmap);
- }
- } else {
- memCache.addBitmapToCache(url, bitmap);
- }
- }
- return bitmap;
- }
七、工作线程池化
有关多线程的切换问题以及在UI线程中执行loadBitmap()方法无效的问题,请参见另一篇博文: 使用严苛模式打破Android4.0以上平台应用中UI主线程的“独断专行”。
有关工作线程的处理方式,这里推荐使用定制线程池的方式,核心代码如下:
- // 线程池初始容量
- private static final int POOL_SIZE = 4;
- private ExecutorService executorService;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- // 获取当前使用设备的CPU个数
- int cpuNums = Runtime.getRuntime().availableProcessors();
- // 预开启线程池数目
- executorService = Executors.newFixedThreadPool(cpuNums * POOL_SIZE);
- ...
- executorService.submit(new Runnable() {
- // 此处执行一些耗时工作,不要涉及UI工作。如果遇到,直接转交UI主线程
- pano.setImage(loadBitmap(url));
- });
- ...
- }
我们知道,线程构造也是比较耗资源的。一定要对其进行有效的管理和维护。千万不要随意而行,一张图片的工作线程不搭理也许没什么,当使用场景变为 ListView和GridView时,线程池化工作就显得尤为重要了。Android不是提供了AsyncTask吗?为什么不用它?其实 AsyncTask底层也是靠线程池支持的,它默认分配的线程数是128,是远大于我们定制的executorService。
转 如何高效使用和管理Bitmap--图片缓存管理模块的设计与实现的更多相关文章
- 如何高效使用和管理Bitmap--图片缓存管理模块的设计与实现
转载请注明 ☞ http://blog.csdn.net/leverage_1229 上周为360全景项目引入了图片缓存模块.因为是在Android4.0平台以上运作,出于惯性,都会在设计之前查阅相关 ...
- [置顶] 如何高效使用和管理Bitmap--图片缓存管理模块的设计与实现
传送门 ☞ 轮子的专栏 ☞ 转载请注明 ☞ http://blog.csdn.net/leverage_1229 上周为360全景项目引入了图片缓存模块.因为是在Android4.0平台以上运作,出于 ...
- (转)Android技术积累:图片缓存管理
如果每次加载同一张图片都要从网络获取,那代价实在太大了.所以同一张图片只要从网络获取一次就够了,然后在本地缓存起来,之后加载同一张图片时就从缓存中加载就可以了.从内存缓存读取图片是最快的,但是因为内存 ...
- android 管理Bitmap内存 - 开发文档翻译
由于本人英文能力实在有限,不足之初敬请谅解 本博客只要没有注明“转”,那么均为原创,转贴请注明本博客链接链接 Managing Bitmap Memory 管理Bitmap内存 In additi ...
- 快速高效实现微信小程序图片上传与腾讯免费5G存储空间的使用
本文介绍了如何在微信小程序开发中使用腾讯官方提供的云开发功能快速实现图片的上传与存储,以及介绍云开发的 5G 存储空间的基本使用方法,这将大大提高微信小程序的开发效率 对于一般的图片上传功能开发,我们 ...
- Android笔记--Bitmap(二)内存管理
Bitmap(二) 内存管理 1.使用内存缓存保证流畅性 这种使用方式在ListView等这种滚动条的展示方式中使用最为广泛, 使用内存缓存 内存缓存位图可以提供最快的展示.但代价就是占用一定的内存空 ...
- Android图片缓存之Bitmap详解
前言: 最近准备研究一下图片缓存框架,基于这个想法觉得还是先了解有关图片缓存的基础知识,今天重点学习一下Bitmap.BitmapFactory这两个类. 图片缓存相关博客地址: Android图片缓 ...
- SDWebImage使用——一个可管理远程图片加载的类库
SDWebImage使用——一个可管理远程图片加载的类库 SDWebImage使用——一个可管理远程图片加载的类库 SDWebImage托管在github上.https://github.co ...
- Android中如何将Bitmap byte裸数据转换成Bitmap图片int数据
Android中如何将Bitmap byte裸数据转换成Bitmap图片int数据 2014-06-11 10:45:14 阅读375次 我们在JNI中处理得到的BMP图片Raw数据,我们应该如何 ...
随机推荐
- laravel安装 笔记
http://laod.cn/hosts/2015-google-hosts.html 谷歌FQIP laravel安装和设置流程 1安装composer , VirtualBox和Vagrant 下 ...
- 不安装oracle客户端,如何运行sqlplus
1.软件下载 http://www.oracle.com/technetwork/topics/linuxx86-64soft-092277.html 下载如下三个包: oracle-instantc ...
- Linux Shell 小脚本经典收藏
原文:http://www.cnblogs.com/Javame/p/3867686.html 1.在两个文件中找出相同的号码 diff -y xx.txt oo.txt | egrep -v &qu ...
- java中的静态变量
大家都知道,我们可以基于一个类创建多个该类的对象,每个对象都拥有自己的成员,互相独立.然而在某些时候,我们更希望该类所有的对象共享同一个成员.此时就是 static 大显身手的时候了!! Java 中 ...
- C++ primer 练习 12.7
重做上一题,这次使用shared_ptr 而不是内置指针.上一题题目为:(编写函数,返回一个动态分配的int的vector.将此vector传递给另一个函数,这个函数读取标准输入,将读入的值保存在ve ...
- button的action属性如果有参数,必须加“:”
比如: [bt addTarget:self action:@selector(shareButtonClickHandler:) …… 后面未写完 一开始我没加,就报错.
- jquery完美实现textarea输入框限制字数
<html> <head> <title> jquery完美实现textarea输入框限制字数</title> <meta http-equiv= ...
- sql关键字过滤C#方法
/// <summary> ///SQL注入过滤 /// </summary> /// <param name="InText">要过滤的字符串 ...
- 转 dos 下的 find 和 重定向
1.find /i "ora-" *.* > check.log 附录: 我对findstr是如此的依赖,以至于当我向各位讲解find命令的时候,我还得老老实实地在cmd窗口 ...
- Toy Storage POJ 2398
题目大意:和 TOY题意一样,但是需要对隔板从左到右进行排序,要求输出的是升序排列的含有i个玩具的方格数,以及i值. 题目思路:判断叉积,二分遍历 #include<iostream> # ...