高效率下载图片——防止内存溢出

在应用中常常须要下载不少的图片,所以,写好图片下载部分的代码很是关键。很差的代码很容易建立太多的对象,致使常常执行GC,接着就出现了ANR;也很容易致使内存溢出OOM。javascript

 

如今,我从防止ANR和OOM的角度写下载图片的代码。再来分析一下需求,当我须要为图片列表下载不少张图片时,我指望图片是有顺序地一张一张显示,而不是开启不少线程同时下载多张图片(注意:这样也会影响每一个线程的执行速度)。java


  1. import java.util.ArrayList;  
  2. import java.util.HashMap;  
  3. import java.util.List;  
  4. import java.util.Map;  
  5.   
  6. import android.graphics.Bitmap;  
  7. import android.graphics.BitmapFactory;  
  8. import android.os.Handler;  
  9. import android.os.Message;  
  10.   
  11. public class ImageDownloadThread extends Thread {  
  12.     //单例类  
  13.     private ImageDownloadThread() {}  
  14.     private static ImageDownloadThread imageDownloadThread = null;  
  15.     public static ImageDownloadThread getInstance() {  
  16.         if (imageDownloadThread == null) {  
  17.             imageDownloadThread = new ImageDownloadThread();  
  18.             imageDownloadThread.start();//建立后马上运行  
  19.         }  
  20.         return imageDownloadThread;  
  21.     }  
  22.       
  23.     //缓存下载图片  
  24.     private Map<String, String> cache = new HashMap<String, String>();//KEY:图片URL;VALUE:下载后的图片路径  
  25.     public boolean isDownload(String imageUrl) {  
  26.         return cache.containsKey(imageUrl);  
  27.     }  
  28.     public Bitmap downloadWithCache(ImageDownloadItem item) {  
  29.         if (cache.containsKey(item.imageUrl)) {  
  30.             Bitmap bitmap = BitmapFactory.decodeFile(cache.get(item.imageUrl));  
  31.             return bitmap;  
  32.         } else {  
  33.             addDownloadItem(item);  
  34.         }  
  35.         return null;  
  36.     }  
  37.     public void downloadWithoutCache(ImageDownloadItem item) {  
  38.         addDownloadItem(item);  
  39.     }  
  40.   
  41.     //下载队列  
  42.     private List<ImageDownloadItem> queue = new ArrayList<ImageDownloadItem>();  
  43.     private synchronized void addDownloadItem(ImageDownloadItem item) {  
  44.         queue.add(item);  
  45.         this.notify();//添加了下载项就激活本线程  
  46.     }  
  47.   
  48.     @Override  
  49.     public void run() {  
  50.         while(true) {  
  51.             while(queue.size() > ) {  
  52.                 ImageDownloadItem item = queue.remove();  
  53.                 String imagePath = downloadImage(item.imageUrl);  
  54.                 //缓存图片路径  
  55.                 cache.put(item.imageUrl, imagePath);  
  56.   
  57.                 if (item.callback != null) {//须要执行回调来显示图片  
  58.                     item.imagePath = imagePath;  
  59.   
  60.                     //交由UI线程处理  
  61.                     Message msg = handler.obtainMessage();  
  62.                     msg.obj = item;  
  63.                     handler.sendMessage(msg);  
  64.                 }  
  65.             }  
  66.             try {  
  67.                 synchronized(this) {  
  68.                     this.wait();//没有下载项时等待  
  69.                 }  
  70.             } catch (InterruptedException e) {  
  71.                 e.printStackTrace();  
  72.             }  
  73.         }  
  74.     }  
  75.       
  76.     private String downloadImage(String imageUrl) {  
  77.         //TODO  
  78.         //不提供该方法代码  
  79.         //下载部分应该有专门下载文件的类(如:FileDownloadUtil.download(imageUrl))  
  80.         return "";  
  81.     }  
  82.   
  83.     private Handler handler = new Handler() {  
  84.         @Override  
  85.         public void handleMessage(Message msg) {  
  86.             ImageDownloadItem item = (ImageDownloadItem)msg.obj;  
  87.             Bitmap bitmap = BitmapFactory.decodeFile(item.imagePath);  
  88.             item.callback.update(bitmap, item.imageUrl);  
  89.         }  
  90.     };  
  91.   
  92.     public static class ImageDownloadItem {  
  93.         public String imageUrl;//须要下载的图片URL  
  94.         public String imagePath;//下载的后图片路径  
  95.         public ImageDownloadCallback callback;//回调方法  
  96.     }  
  97.       
  98.     public static interface ImageDownloadCallback {  
  99.         //策略模式,由子类实现  
  100.         public void update(Bitmap bitmap, String imageUrl);  
  101.     }  
  102. }  

 

下面是使用的代码片断android

 

Java代码    收藏代码
  1. public View getView(int position, View convertView, ViewGroup vg) {  
  2.         final ImageView imageView;  
  3.         if (convertView != null) {  
  4.             imageView = (ImageView)convertView;  
  5.         } else {  
  6.             imageView = new ImageView(this);  
  7.         }  
  8.         //在实际应用中imageUrl值是不一样的  
  9.         String imageUrl = "http://www.nxnet.net/yule/yljj/200710/W020071008388975463611.jpg";  
  10.         imageView.setTag(imageUrl);  
  11.           
  12.         //设置下载项  
  13.         ImageDownloadItem item = new ImageDownloadItem();  
  14.         item.imageUrl = imageUrl;  
  15.         //若是是无需显示图片的状况(如预下载),无需设置item.callback,即让item.callback = null  
  16.         item.callback = new ImageDownloadCallback() {  
  17.             @Override  
  18.             public void update(Bitmap bitmap, String imageUrl) {  
  19.                 ImageView imageViewByTag = (ImageView)imageView.findViewWithTag(imageUrl);  
  20.                 if (imageViewByTag != null) imageViewByTag.setImageBitmap(bitmap);  
  21.             }  
  22.         };  
  23.           
  24.         ImageDownloadThread imageDownloadThread = ImageDownloadThread.getInstance();  
  25.         Bitmap bitmap = imageDownloadThread.downloadWithCache(item);  
  26.         if (bitmap != null) {//从缓存中取到  
  27.             imageView.setImageBitmap(bitmap);  
  28.         }  
  29.         return imageView;  
  30.     }  
相关文章
相关标签/搜索