android里图片下载工具类AsyncImageLoader分析

时间看见很多人做listview(比如模拟新浪客户端)用到这一个图片下载的类,我也不知道这个类到底是哪个大神写的,反正我使用这个类的时候接手别人的,刚开始,感觉这个类写的听不错,比我写的AsyncImageTask.java好多了,先说说我最开始写的吧,也算是抛砖引玉:
复制代码
public class AsyncImageTask extends AsyncTask<String, Void, InputStream>{
    private ImageView imageView;
    public AsyncImageTask(ImageView imageView){
        this.imageView = imageView;
    }
    @Override
    protected InputStream doInBackground(String... params) {        

        InputStream inputStream = null;
        try {
            URL url = new URL(params[0]);
            inputStream = url.openStream();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return inputStream;
    }
    @Override
    protected void onPostExecute(InputStream result) {
        if(imageView!=null && result!=null){
            Bitmap bmp = BitmapFactory.decodeStream(result);
            imageView.setImageBitmap(bmp);
        }
        super.onPostExecute(result);
    }
}
复制代码

我写完的时候感觉就挺别扭的的,能满足我当时的需求,也就这么用了(有哪位高手指点其中的缺点哈),这个类里,我们只需要传入显示图片的ImageView和图片的网络地址,当图片下载完成后,显示出来就可以了。后来我的主管把它换掉了,用了AsyncImageLoader.java这个类,仔细看看,写的确实不错。不说废话,先把这个类的代码贴上去(AsyncImageLoader.java):

  

复制代码
public class AsyncImageLoader {

     private HashMap<String, SoftReference<Drawable>> imageCache;
      
         public AsyncImageLoader() {
             imageCache = new HashMap<String, SoftReference<Drawable>>();
         }
      
         public Drawable loadDrawable(final String imageUrl, final ImageCallback imageCallback) {
             if (imageCache.containsKey(imageUrl)) {
                 SoftReference<Drawable> softReference = imageCache.get(imageUrl);
                 Drawable drawable = softReference.get();
                 if (drawable != null) {
                     return drawable;
                 }
             }
             final Handler handler = new Handler() {
                 public void handleMessage(Message message) {
                     imageCallback.imageLoaded((Drawable) message.obj, imageUrl);
                 }
             };
             new Thread() {
                 @Override
                 public void run() {
                     Drawable drawable = loadImageFromUrl(imageUrl);
                     imageCache.put(imageUrl, new SoftReference<Drawable>(drawable));
                     Message message = handler.obtainMessage(0, drawable);
                     handler.sendMessage(message);
                 }
             }.start();
             return null;
         }
      
        public static Drawable loadImageFromUrl(String url) {
            URL m;
            InputStream i = null;
            try {
                m = new URL(url);
                i = (InputStream) m.getContent();
            } catch (MalformedURLException e1) {
                e1.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }
            Drawable d = Drawable.createFromStream(i, "src");
            return d;
        }
      
         public interface ImageCallback {
             public void imageLoaded(Drawable imageDrawable, String imageUrl);
         }

}
复制代码

  实现方式:通过传入图片的网络地址和一个实现ImageCallback行为的对象,当imageCache存在这个图片时候,返回这个图片,当imageCache没有这个图片时,实例一个异步线程来下载图片并同时返回为null,最后在图片下载完成的时候,调用imageLoaded方法。

  现说说这个类设计的优点吧:1.采用了策略模式;2.使用SoftReference关键字

  先说说策略模式,程序里把每次下载图片完成后所进行的操作封装成一个ImageCallback抽象类,使系统更灵活,并易于扩展

  在Java中内存管理,引用分为四大类,强引用HardReference、弱引用WeakReference、软引用SoftReference和虚引用PhantomReference。它们的区别也很明显,HardReference对象是即使虚拟机内存吃紧抛出OOM也不会导致这一引用的对象被回收,而WeakReference等更适合于一些数量不多,但体积稍微庞大的对象,在这四个引用中,它是最容易被垃圾回收的,而我们对于显示类似Android Market中每个应用的App Icon时可以考虑使用SoftReference来解决内存不至于快速回收,同时当内存短缺面临Java VM崩溃抛出OOM前时,软引用将会强制回收内存,最后的虚引用一般没有实际意义,仅仅观察GC的活动状态,对于测试比较实用同时必须和ReferenceQueue一起使用。对于一组数据,我们可以通过HashMap的方式来添加一组SoftReference对象来临时保留一些数据,同时对于需要反复通过网络获取的不经常改变的内容,可以通过本地的文件系统或数据库来存储缓存。

  最后一句话说的很对,事实上大多数情况也是如此。

  在说说它的用法吧,通常它作为一个adapter的一个变量如:

复制代码
class BookAdapter extends ArrayAdapter<BookInfo>{
        AsyncImageLoader asyncImageLoader;
        Context mContext;
        
        BookAdapter(Context context,List<BookInfo> data){
            super(context, 0, data);
            asyncImageLoader = new AsyncImageLoader();
            mContext = context;
        }
        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            ViewCache holder ;
            if(convertView==null){
                LayoutInflater inflate = (LayoutInflater)mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                convertView = inflate.inflate(com.slider.cn.R.layout.list_item , null);
                holder = new ViewCache();
                holder.icon = (ImageView)convertView.findViewById(com.slider.cn.R.id.note_icon);
                holder.name = (TextView)convertView.findViewById(com.slider.cn.R.id.note_name);
                holder.date = (TextView)convertView.findViewById(com.slider.cn.R.id.note_date);
                convertView.setTag(holder);
            }else{
                holder = (ViewCache)convertView.getTag();
            }
            final BookInfo bookInfo = getItem(position);
            holder.name.setText(bookInfo.getName().toString());
            holder.date.setText(bookInfo.getInfo());
            holder.icon.setTag(bookInfo.getUri());
            //
            Drawable drawable = asyncImageLoader.loadDrawable(bookInfo.getUri(), new ImageCallback() {
                
                @Override
                public void imageLoaded(Drawable imageDrawable, String imageUrl) {
                    ImageView imageViewByTag = (ImageView) BookListView.this.findViewWithTag(bookInfo.getUri());
                    if (imageViewByTag!=null) {
                        imageViewByTag.setImageDrawable(imageDrawable);
                    }else {
                        //load image failed from Internet
                    }
                    
                }
            });
            if(drawable==null){
                holder.icon.setImageDrawable(drawable_waiting);
            }else{
                
                holder.icon.setImageDrawable(drawable);
            }
            return convertView;
        }
    }
    static class ViewCache{
        ImageView icon;
        TextView name;
        TextView date;
    }
复制代码

   但是,它好像也有一些不完美的地方,比如说可能会造成同时下载二十多个图片的线程(甚至更多),它没有对线程的数量做一个限制。那就使用固定数据的线程池吧,再比如出现重复加在一个图片怎么处理,再比如线程池里线程的优先级安排怎么弄呢?(比如你想要最近添加进入的线程拥有的优先级最高,因为你总是想最先看到当前的界面的内容,而不在乎跳过界面的内容什么时候加在完毕,这里可以说的就太多了,事实上完成上面的已经可以应付大多数应用了)

 

 

 

 

 

--------------------------------------------------------

ListView 是一种可以显示一系列项目并能进行滚动显示的 View,每一行的Item可能包含复杂的结构,可能会从网络上获取icon等的一些图标信息,就现在的网络速度要想保持ListView运行的很好滚动流畅是做不到的

 

所以这里就需要把这些信息利用多线程实现异步加载

 

实现这样功能的类:

 

Java代码    收藏代码
  1. public class AsyncImageLoader {  
  2.     private HashMap<String, SoftReference<Drawable>> imageCache;  
  3.   
  4.     public AsyncImageLoader() {  
  5.         this.imageCache = new HashMap<String, SoftReference<Drawable>>();  
  6.     }  
  7.   
  8.     // 下载图片  
  9.     public static Drawable loadImageFromUrl(String url) {  
  10.         InputStream localInputStream = null;  
  11.         try {  
  12.             localInputStream = (InputStream) new URL(url).getContent();  
  13.             Drawable localDrawable = null;  
  14.             if (localInputStream != null)  
  15.                 localDrawable = Drawable.createFromStream(localInputStream,  
  16.                         "src");  
  17.             return localDrawable;  
  18.         } catch (Exception localException) {  
  19.             localException.printStackTrace();  
  20.             return null;  
  21.         }  
  22.     }  
  23.   
  24.     public HashMap<String, SoftReference<Drawable>> getImageCache() {  
  25.         return imageCache;  
  26.     }  
  27.   
  28.     public void setImageCache(  
  29.             HashMap<String, SoftReference<Drawable>> imageCache) {  
  30.         this.imageCache = imageCache;  
  31.     }  
  32.   
  33.     @SuppressWarnings("unchecked")  
  34.     public Drawable loadDrawable(String imageURL,  
  35.             ImageCallback paramImageCallback) {  
  36.   
  37.         boolean bool = this.imageCache.containsKey(imageURL);  
  38.         Drawable localDrawable = null;  
  39.         if (bool) {  
  40.             localDrawable = (Drawable) ((SoftReference) (imageCache)  
  41.                     .get(imageURL)).get();  
  42.             return localDrawable;  
  43.         } else {  
  44.             ImageHandler imageHandler = new ImageHandler(this,  
  45.                     paramImageCallback, imageURL);  
  46.             new ImageThread(this, imageURL, imageHandler).start();  
  47.         }  
  48.         return localDrawable;  
  49.     }  
  50.   
  51. public abstract interface ImageCallback {  
  52.         public abstract void imageLoaded(Drawable drawable, String imageUrl);  
  53.     }  
  54.   
  55.     class ImageHandler extends Handler {  
  56.         AsyncImageLoader asyncImageLoader;  
  57.         ImageCallback imageCallback;  
  58.         String imageUrl;  
  59.   
  60.         public ImageHandler(AsyncImageLoader asyncImageLoader,  
  61.                 ImageCallback paramImageCallback, String paramString) {  
  62.             this.asyncImageLoader = asyncImageLoader;  
  63.             this.imageCallback = paramImageCallback;  
  64.             this.imageUrl = paramString;  
  65.         }  
  66.   
  67.         public void handleMessage(Message paramMessage) {  
  68.             if (this.imageCallback != null) {  
  69.                 Drawable localDrawable = (Drawable) paramMessage.obj;  
  70.                 // 将图片和图片的URL 传给回调函数,在回调函数中进行相应操作。  
  71.                 imageCallback.imageLoaded(localDrawable, this.imageUrl);  
  72.             }  
  73.         }  
  74.     }  
  75.   
  76.     class ImageThread extends Thread {  
  77.         private AsyncImageLoader asyncImageLoader;  
  78.         private String imageURL;  
  79.         private ImageHandler imageHandler;  
  80.   
  81.         public ImageThread(AsyncImageLoader asyncImageLoader, String imageURL,  
  82.                 ImageHandler imageHandler) {  
  83.             this.asyncImageLoader = asyncImageLoader;  
  84.             this.imageURL = imageURL;  
  85.             this.imageHandler = imageHandler;  
  86.         }  
  87.   
  88.         public void run() {  
  89.             Drawable localDrawable = AsyncImageLoader  
  90.                     .loadImageFromUrl(this.imageURL);  
  91.             // 将新下载的图片存入imageCache  
  92.             HashMap imageCache = asyncImageLoader.getImageCache();  
  93.             String str = this.imageURL;  
  94.             SoftReference localSoftReference = new SoftReference(localDrawable);  
  95.             imageCache.put(str, localSoftReference);  
  96.             Message localMessage = this.imageHandler.obtainMessage(0,  
  97.                     localDrawable);  
  98.             // 将图片 通过消息发送给hander  
  99.             this.imageHandler.sendMessage(localMessage);  
  100.         }  
  101.     }  
  102. }  

 注意这里使用了 SoftReference来缓存图片,允许 GC在需要的时候可以对缓存中的图片进行清理。它这样工作:

 

<!--[if !supportLists]-->l         <!--[endif]-->调用 loadDrawable(ImageUrl, imageCallback),传入一个匿名实现的 ImageCallback接口

<!--[if !supportLists]-->l         <!--[endif]-->如果图片在缓存中不存在的话,图片将从单一的线程中下载并在下载结束时通过 ImageCallback回调

<!--[if !supportLists]-->l         <!--[endif]-->如果图片确实存在于缓存中,就会马上返回,不会回调 ImageCallback

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值