`
hunankeda110
  • 浏览: 741868 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

listView 图片加载性能优化

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



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



实现这样功能的类





[java] view plaincopyprint?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() { 
            @Override 
            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) { 
        // ...  
    } 
  
    public interface ImageCallback { 
        public void imageLoaded(Drawable imageDrawable, String imageUrl); 
    } 

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() {
            @Override
            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) {
        // ...
    }

    public interface ImageCallback {
        public void imageLoaded(Drawable imageDrawable, String imageUrl);
    }
}




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

·         调用 loadDrawable(ImageUrl, imageCallback),传入一个匿名实现的 ImageCallback接口

·         如果图片在缓存中不存在的话,图片将从单一的线程中下载并在下载结束时通过 ImageCallback回调

·         如果图片确实存在于缓存中,就会马上返回,不会回调 ImageCallback



        然后我们还可以根据09google I/0开发者大会提到的方式来继续优化Adapter 使用ViewHolder来减少一些比较费时的操作,譬如inflate XML 和 findViewById()等操作

       [java] view plaincopyprint?public class ImageAndTextListAdapter extends ArrayAdapter<ImageAndText> { 
  
    private ListView listView; 
    private AsyncImageLoader asyncImageLoader; 
  
    public ImageAndTextListAdapter(Activity activity, List<ImageAndText> imageAndTexts, ListView listView) { 
        super(activity, 0, imageAndTexts); 
        this.listView = listView; 
        asyncImageLoader = new AsyncImageLoader(); 
    } 
  
    @Override 
    public View getView(int position, View convertView, ViewGroup parent) { 
        Activity activity = (Activity) getContext(); 
  
        // Inflate the views from XML  
        View rowView = convertView; 
        ViewCache viewCache; 
        if (rowView == null) { 
            LayoutInflater inflater = activity.getLayoutInflater(); 
            rowView = inflater.inflate(R.layout.image_and_text_row, null); 
            viewCache = new ViewCache(rowView); 
            rowView.setTag(viewCache); 
        } else { 
            viewCache = (ViewCache) rowView.getTag(); 
        } 
        ImageAndText imageAndText = getItem(position); 
  
        // Load the image and set it on the ImageView  
        String imageUrl = imageAndText.getImageUrl(); 
        ImageView imageView = viewCache.getImageView(); 
        imageView.setTag(imageUrl); 
        Drawable cachedImage = asyncImageLoader.loadDrawable(imageUrl, new ImageCallback() { 
            public void imageLoaded(Drawable imageDrawable, String imageUrl) { 
                ImageView imageViewByTag = (ImageView) listView.findViewWithTag(imageUrl); 
                if (imageViewByTag != null) { 
                    imageViewByTag.setImageDrawable(imageDrawable); 
                } 
            } 
        }); 
        imageView.setImageDrawable(cachedImage); 
  
        // Set the text on the TextView  
        TextView textView = viewCache.getTextView(); 
        textView.setText(imageAndText.getText()); 
  
        return rowView; 
    } 
}  
public class ImageAndTextListAdapter extends ArrayAdapter<ImageAndText> {

    private ListView listView;
    private AsyncImageLoader asyncImageLoader;

    public ImageAndTextListAdapter(Activity activity, List<ImageAndText> imageAndTexts, ListView listView) {
        super(activity, 0, imageAndTexts);
        this.listView = listView;
        asyncImageLoader = new AsyncImageLoader();
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        Activity activity = (Activity) getContext();

        // Inflate the views from XML
        View rowView = convertView;
        ViewCache viewCache;
        if (rowView == null) {
            LayoutInflater inflater = activity.getLayoutInflater();
            rowView = inflater.inflate(R.layout.image_and_text_row, null);
            viewCache = new ViewCache(rowView);
            rowView.setTag(viewCache);
        } else {
            viewCache = (ViewCache) rowView.getTag();
        }
        ImageAndText imageAndText = getItem(position);

        // Load the image and set it on the ImageView
        String imageUrl = imageAndText.getImageUrl();
        ImageView imageView = viewCache.getImageView();
        imageView.setTag(imageUrl);
        Drawable cachedImage = asyncImageLoader.loadDrawable(imageUrl, new ImageCallback() {
            public void imageLoaded(Drawable imageDrawable, String imageUrl) {
                ImageView imageViewByTag = (ImageView) listView.findViewWithTag(imageUrl);
                if (imageViewByTag != null) {
                    imageViewByTag.setImageDrawable(imageDrawable);
                }
            }
        });
        imageView.setImageDrawable(cachedImage);

        // Set the text on the TextView
        TextView textView = viewCache.getTextView();
        textView.setText(imageAndText.getText());

        return rowView;
    }






      这里我们没有加载完iamge之后直接设定到相应的ImageView上 ,而是通过Tag查找,这里我们重用的View 这里有个listView的引用来通过Tag查找 可见 CallBack的实现



      [c-sharp] view plaincopyprint?ImageView imageViewByTag = (ImageView) listView.findViewWithTag(imageUrl); 
               if (imageViewByTag != null) { 
                   imageViewByTag.setImageDrawable(imageDrawable); 
               } 
ImageView imageViewByTag = (ImageView) listView.findViewWithTag(imageUrl);
                if (imageViewByTag != null) {
                    imageViewByTag.setImageDrawable(imageDrawable);
                }



      这里通过ViewCatch来减少了 findViewById的使用



     [c-sharp] view plaincopyprint?public class ViewCache { 
  
    private View baseView; 
    private TextView textView; 
    private ImageView imageView; 
  
    public ViewCache(View baseView) { 
        this.baseView = baseView; 
    } 
  
    public TextView getTextView() { 
        if (textView == null) { 
            textView = (TextView) baseView.findViewById(R.id.text); 
        } 
        return titleView; 
    } 
  
    public ImageView getImageView() { 
        if (imageView == null) { 
            imageView = (ImageView) baseView.findViewById(R.id.image); 
        } 
        return imageView; 
    } 
}  
public class ViewCache {

    private View baseView;
    private TextView textView;
    private ImageView imageView;

    public ViewCache(View baseView) {
        this.baseView = baseView;
    }

    public TextView getTextView() {
        if (textView == null) {
            textView = (TextView) baseView.findViewById(R.id.text);
        }
        return titleView;
    }

    public ImageView getImageView() {
        if (imageView == null) {
            imageView = (ImageView) baseView.findViewById(R.id.image);
        }
        return imageView;
    }




     总结 :这里主要做了三点优化





在单一线程里加载图片
  重用列表中行
缓存行中的 View
分享到:
评论

相关推荐

    ListView性能优化之视图缓存

    ListView性能优化之视图缓存

    listview异步加载优化实例

    在APP应用中,listview的异步加载图片方式能够带来很好的用户体验,同时也是考量程序性能的一个重要指标。关于listview的异步加载,网上其实很多示例了,中心思想都差不多,不过很多版本或是有bug,或是有性能问题...

    Listview加载的性能优化是如何实现的

    在android开发中Listview是一个很重要的组件,它以列表的形式根据数据的长自适应展示具体内容,用户可以自由的定义listview每一列的布局,接下来通过本文给大家介绍Listview加载的性能优化是如何实现的,对listview...

    Listview的异步加载性能优化

    关于listview的异步加载,网上其实很多示例了,中心思想都差不多,不过很多版本或是有bug,或是有性能问题有待优化,下面就让在下阐述其原理以探索个中奥秘在APP应用中,listview的异步加载图片方式能够带来很好的...

    android listview 加载网络图片(线程池+缓存管理) 高性能

    android开发必备知识,android listview 加载网络图片 采用 可管理线程池+文件缓存+网络优化 欢迎下载参考

    ListView加载多item布局

    ListView加载多item布局及性能加载优化

    ListView显示大量图片时的性能优化

    1)采用仅显示当前有效内容的方式缩小用户响应时间; 2)采用定时器后台加载的方式; 2000个资源的显示时间可控制在1s左右。

    ListView性能优化

    数据适配性能优化;listview的快速拖动的问题;listview的延时加载。

    ListView异步加载图片实现思路(优化篇)

    关于listview的异步加载,网上其实很多示例了,中心思想都差不多,不过很多版本或是有bug,或是有性能问题有待优化,下面就让在下阐述其原理以探索个中奥秘

    Android代码-实现了在列表控件(ListView, RecyclerView)中加载并播放视频

    VideoListPlayer ...自动播放/停止功能性能优化 视频播放加入声音开关控制,默认播放视频关闭声音,点击视频开启声音 fix在4.1.1以下无法播放视频的bug 基本用法 添加gradle依赖 repositories { maven { ur

    ListView的简单应用和滑动加载数据

    ListView这个控件在Android开发中用到非常频繁,基本上是个应用都会到,它以列表的形式来展示一些数据,本篇博客只是一个是简单实例,来帮助初学者理解ListView,不涉及到性能优化。

    浅谈Android开发中ListView控件性能的一些优化方法

    ListView优化一直是一个老生常谈的问题,不管是面试还是平常的开发中,ListView永远不会被忽略掉,那么这篇文章我们来看看如何最大化的优化ListView的性能。 1.在adapter中的getView方法中尽量少使用逻辑 2.尽最大...

    Android编程开发之性能优化技巧总结

    2.listview 性能优化 1).复用convertView 在getItemView中,判断convertView是否为空,如果不为空,可复用。如果couvertview中的view需要添加listerner,代码一定要在if(convertView==null){}之外。 2).异步加载图片...

    Android代码-处理异步下载的库

    优化了listview等快速滑动时的图片加载 圆角图采用了性能最优的方案 可设置模糊效果和头像识别自动剪裁功能 通过各种手段保障gridview等在有圆角和大量gif的情况下快速滑动时也能依旧极度流畅。 通过自定义的方式...

    react-native-easy-listview-gridview:React Native中ListView和GridView的包装器库

    样本列表显示网格视图特征拉刷新到达列表视图或网格视图的末尾时加载更多内容为空时列表视图或网格视图的自定义背景自定义样式的RefreshControl 自定义页眉,页脚,分隔符等TIMEOUT加载处理程序性能优化更多即将推出...

    超实用的jQuery代码段

    7.20 如何判断加载多张图片的完成状态 7.21 鼠标悬停时的图片放大 7.22 淡出图片,淡入另一幅图片 7.23 页面加载时随机显示图片 7.24 按顺序淡入图片显示 7.25 检测图片的URL是否有效 7.26 强制显示图片的方法 7.27 ...

    Android开发艺术探索.任玉刚(带详细书签).pdf

    第三,介绍一些核心技术和Android的性能优化思想。 第1章 Activity的生命周期和启动模式 1 1.1 Activity的生命周期全面分析 1 1.1.1 典型情况下的生命周期分析 2 1.1.2 异常情况下的生命周期分析 8 1.2 ...

Global site tag (gtag.js) - Google Analytics