android异步图片加载三之handler+线程池+消息队列模式+缓存

简介:

http://blog.csdn.net/soesa/article/details/7787922

假象现在有这样一个需求:

      一个微博客户端或者多媒体客户端需要从服务端拉取一些数据,该数据中包含了很多图片,如果不用异步加载方式,可能会使主线程阻塞导致ANR异常,但是异步加载后如果每次都通过网络去下载图片,性能上会差很多,而且也会浪费流量,导致用户的不满,故现在增加一个缓存用来存储图片,缓存中包含内存存储和sdcard存储。先从缓存中去,在缓存中先从内存中取出,如果内存中没有再从sdcard去,如果sdcard中存在则直接返回该图片资源并放入内存中,如果两者都没有则从网络上下载,下载后并放入sdcard中供下次使用,这样就节约了资源开销,下面是主要代码:

PicManager(用来处理图片的操作):

[java]  view plain copy
  1. public class PicManager {  
  2.   
  3.     private Map<String,SoftReference<Bitmap>> imgCache ;  
  4.     private Context ctx;  
  5.       
  6.     public PicManager(Map<String,SoftReference<Bitmap>> imgCache,Context ctx) {  
  7.         this.ctx = ctx;  
  8.         this.imgCache = imgCache;  
  9.     }  
  10.       
  11.       
  12.     /** 
  13.      * 从网络上下载 
  14.      * @param url 
  15.      * @return 
  16.      */  
  17.     public Bitmap getBitMapFromUrl(String url) {  
  18.         Bitmap bitmap = null;  
  19.         URL u =null;  
  20.         HttpURLConnection conn = null;  
  21.         InputStream is = null;  
  22.         try {  
  23.             u = new URL(url);  
  24.             conn = (HttpURLConnection)u.openConnection();  
  25.             is = conn.getInputStream();  
  26.             bitmap = BitmapFactory.decodeStream(is);  
  27.         } catch (Exception e) {  
  28.             e.printStackTrace();  
  29.         }  
  30.         return bitmap;  
  31.     }  
  32.       
  33.     /** 
  34.      * 从文件中读取 
  35.      * @return 
  36.      * @throws Exception  
  37.      */  
  38.     private Bitmap getBitMapFromSDCard(String url) throws Exception {  
  39.         Bitmap bitmap = null;  
  40.         String filename = MD5Util.getMD5(url);  
  41.         FileInputStream fis = ctx.openFileInput(filename);  
  42.         bitmap = BitmapFactory.decodeStream(fis);  
  43.         return bitmap;  
  44.     }  
  45.       
  46.     /** 
  47.      * 从缓存中读取 
  48.      * @param url 
  49.      * @return 
  50.      * @throws Exception  
  51.      */  
  52.     public Bitmap getImgFromCache(String url) throws Exception {  
  53.         Bitmap bitmap = null;  
  54.         //从内存中读取  
  55.         if(imgCache.containsKey(url)) {  
  56.             synchronized (imgCache) {  
  57.                 SoftReference<Bitmap> bitmapReference = imgCache.get(url);  
  58.                 if(null != bitmapReference) {  
  59.                     bitmap = bitmapReference.get();  
  60.                 }  
  61.             }  
  62.         } else {//否则从文件中读取  
  63.             bitmap = getBitMapFromSDCard(url);  
  64.             //将图片保存进内存中  
  65.             imgCache.put(url, new SoftReference<Bitmap>(bitmap));  
  66.         }  
  67.         return bitmap;  
  68.     }  
  69.       
  70.     /** 
  71.      * 将图片写入sdcard中 
  72.      * @param bitmap 
  73.      * @param url 
  74.      * @throws Exception 
  75.      */  
  76.     public void writePic2SDCard(Bitmap bitmap,String url) throws Exception {  
  77.         String filename = MD5Util.getMD5(url);  
  78.         FileOutputStream fos = ctx.openFileOutput(filename, ctx.MODE_APPEND);  
  79.         byte[] bitmapByte = PictureUtil.bitmap2Byte(bitmap);  
  80.         ByteArrayInputStream bis = new ByteArrayInputStream(bitmapByte);  
  81.         int len = 0;  
  82.         byte[] b = new byte[bis.available()];  
  83.         while((len = bis.read(b)) != -1) {  
  84.             fos.write(b, 0, len);  
  85.         }  
  86.         if(null != bis) {  
  87.             bis.close();  
  88.         }  
  89.         if(null != fos) {  
  90.             fos.close();  
  91.         }  
  92.     }  
  93. }  

 ImgLoader(用来加在图片):


[java]  view plain copy
  1. public class ImgLoader {  
  2.   
  3.     private Map<String,SoftReference<Bitmap>> imgCache = new HashMap<String,SoftReference<Bitmap>>();  
  4.     private Context ctx;  
  5.       
  6.     public ImgLoader(Context ctx) {  
  7.         this.ctx = ctx;  
  8.     }  
  9.       
  10.     private PicManager manager = new PicManager(imgCache,ctx);  
  11.     private Handler handler = new Handler();  
  12.       
  13.       
  14.     private ExecutorService threadPool = Executors.newFixedThreadPool(5);  
  15.       
  16.     public Bitmap loadImg(final String url,final ImgCallback callback) {  
  17.           
  18.         //先从缓存中读取图片资源  
  19.         Bitmap bitmap = null;  
  20.         try {  
  21.             bitmap = manager.getImgFromCache(url);;  
  22.               
  23.             if(null == bitmap) {  
  24.                 //开启线程从网络上下载  
  25.                 threadPool.submit(new Runnable() {//submit方法确保下载是从线程池中的线程执行  
  26.                     @Override  
  27.                     public void run() {  
  28.                         final Bitmap bitmapFromUrl = manager.getBitMapFromUrl(url);  
  29.                         try {  
  30.                             manager.writePic2SDCard(bitmapFromUrl,url);  
  31.                         } catch (Exception e) {  
  32.                             e.printStackTrace();  
  33.                         }  
  34.                         handler.post(new Runnable() {  
  35.                             @Override  
  36.                             public void run() {  
  37.                                 callback.refresh(bitmapFromUrl);  
  38.                             }  
  39.                         });  
  40.                     }  
  41.                 });  
  42.             } else {  
  43.                   
  44.             }  
  45.         } catch (Exception e) {  
  46.             e.printStackTrace();  
  47.         }  
  48.         return bitmap;  
  49.     }  
  50.       
  51. }  

 ImgCallback callback是一个接口,用来刷新界面


[java]  view plain copy
  1. public interface ImgCallback {  
  2.   
  3.       
  4.     public void refresh(Bitmap bitmap);  
  5. }  

相关文章
|
7月前
|
canal 缓存 NoSQL
Redis缓存与数据库如何保证一致性?同步删除+延时双删+异步监听+多重保障方案
根据对一致性的要求程度,提出多种解决方案:同步删除、同步删除+可靠消息、延时双删、异步监听+可靠消息、多重保障方案
Redis缓存与数据库如何保证一致性?同步删除+延时双删+异步监听+多重保障方案
|
2月前
|
SQL 数据建模 BI
【YashanDB 知识库】用 yasldr 配置 Bulkload 模式作单线程迁移 300G 的业务数据到分布式数据库,迁移任务频繁出错
问题描述 详细版本:YashanDB Server Enterprise Edition Release 23.2.4.100 x86_64 6db1237 影响范围: 离线数据迁移场景,影响业务数据入库。 外场将部分 NewCIS 的报表业务放到分布式数据库,验证 SQL 性能水平。 操作系统环境配置: 125G 内存 32C CPU 2T 的 HDD 磁盘 问题出现的步骤/操作: 1、部署崖山分布式数据库 1mm 1cn 3dn 单线启动 yasldr 数据迁移任务,设置 32 线程的 bulk load 模式 2、观察 yasldr.log 是否出现如下错
|
8月前
|
缓存 安全 Android开发
Android经典实战之用Kotlin泛型实现键值对缓存
本文介绍了Kotlin中泛型的基础知识与实际应用。泛型能提升代码的重用性、类型安全及可读性。文中详细解释了泛型的基本语法、泛型函数、泛型约束以及协变和逆变的概念,并通过一个数据缓存系统的实例展示了泛型的强大功能。
76 2
|
6月前
|
缓存 Java Shell
Android 系统缓存扫描与清理方法分析
Android 系统缓存从原理探索到实现。
192 15
Android 系统缓存扫描与清理方法分析
|
4月前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
5月前
|
API Android开发 iOS开发
深入探索Android与iOS的多线程编程差异
在移动应用开发领域,多线程编程是提高应用性能和响应性的关键。本文将对比分析Android和iOS两大平台在多线程处理上的不同实现机制,探讨它们各自的优势与局限性,并通过实例展示如何在这两个平台上进行有效的多线程编程。通过深入了解这些差异,开发者可以更好地选择适合自己项目需求的技术和策略,从而优化应用的性能和用户体验。
|
4月前
|
缓存 安全 Java
【JavaEE】——单例模式引起的多线程安全问题:“饿汉/懒汉”模式,及解决思路和方法(面试高频)
单例模式下,“饿汉模式”,“懒汉模式”,单例模式下引起的线程安全问题,解锁思路和解决方法
|
7月前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android多线程编程的重要性及其实现方法,涵盖了基本概念、常见线程类型(如主线程、工作线程)以及多种多线程实现方式(如`Thread`、`HandlerThread`、`Executors`、Kotlin协程等)。通过合理的多线程管理,可大幅提升应用性能和用户体验。
234 15
一个Android App最少有几个线程?实现多线程的方式有哪些?
|
6月前
|
调度 Android开发 开发者
构建高效Android应用:探究Kotlin多线程优化策略
【10月更文挑战第11天】本文探讨了如何在Kotlin中实现高效的多线程方案,特别是在Android应用开发中。通过介绍Kotlin协程的基础知识、异步数据加载的实际案例,以及合理使用不同调度器的方法,帮助开发者提升应用性能和用户体验。
117 4