[Android]ListView的Adapter.getView()方法中延迟加载图片的优化

转载 2016年05月30日 14:59:10

个例子吧,以好友列表为例

ListView中每个Item表示一个好友,每个好友中都有一个头像,需要从服务端加载到本地,然后显示在item中。

显然,启动加载图片的过程应该是在getView()方法中触发,启动一个线程,然后下载头像图片。这里使用我写的一个开源框架ImageLoaderSamplehttps://github.com/wangjiegulu/ImageLoaderSample)来加载图片,并实现内存缓存和本地缓存。

额--这里不再介绍ImageLoaderSample的用法了,给个传送门:http://www.cnblogs.com/tiantianbyconan/p/3574131.html

 

再来看看getView()方法的调用时机:

1. Adapter调用NotifyDataChanged的时候

2. ListView滚动时,也就是convertView不断复用的时候。

也就是说,每当ListView滚动时,getView()方法不断被调用,图片下载的过程不断地执行(当然,ImageLoaderSample中会有缓存,但是内存缓存时有限的,如果内存缓存中找不到要显示的图片,那就需要到文件缓存中查找,需要进行io读写,这个也是相对比较耗时的),显然,这里面还有优化的余地。

怎么去优化这里?只要让ListView滚动的时候图片显示的时候不要去进行io读写就好了,具体逻辑如下:

-如果调用GetView方法时,ListView处于停止状态,则先去内存中查找头像图片;如果内存图片存在,则显示内存中保存好的图片;如果内存图片不存在,则继续到文件缓存中找,如果文件缓存图片存在,则显示文件缓存中的图片;如果文件缓存图片不存在,则根据url去网络下载这张图片,然后显示;

-如果调用getView方法时,ListView处于滚动状态,则去内存中查找头像的图片;如果内存图片存在,则显示内存中保存好的图片;如果内存图片不存在,则显示一张默认的图片(省去了从文件缓存中找图片和网络中去请求图片的步骤)。

 

这样的话,我们就必须要改写BaseAdapter,让它能够监测ListView的滚动状态,并在Adapter中可以获取到当前ListView的滚动状态。所以改造BaseAdapter,ABaseAdapterhttps://github.com/wangjiegulu/AndroidBucket/blob/master/src/com/wangjie/androidbucket/adapter/ABaseAdapter.java):

复制代码
 1 package com.wangjie.androidbucket.adapter;
 2 
 3 import android.widget.*;
 4 import com.wangjie.androidbucket.adapter.listener.OnAdapterScrollListener;
 5 
 6 /**
 7  * Author: wangjie
 8  * Email: tiantian.china.2@gmail.com
 9  * Date: 12/3/14.
10  */
11 public abstract class ABaseAdapter extends BaseAdapter implements AbsListView.OnScrollListener {
12     private OnAdapterScrollListener onAdapterScrollListener;
13     /**
14      * 当前listview是否属于滚动状态
15      */
16     private boolean isScrolling;
17 
18     public boolean isScrolling() {
19         return isScrolling;
20     }
21 
22     public void setOnAdapterScrollListener(OnAdapterScrollListener onAdapterScrollListener) {
23         this.onAdapterScrollListener = onAdapterScrollListener;
24     }
25 
26     protected ABaseAdapter(AbsListView listView) {
27         listView.setOnScrollListener(this);
28     }
29 
30     @Override
31     public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
32         if (null != onAdapterScrollListener) {
33             onAdapterScrollListener.onScroll(view, firstVisibleItem, visibleItemCount, totalItemCount);
34         }
35     }
36 
37     @Override
38     public void onScrollStateChanged(AbsListView view, int scrollState) {
39         if (null != onAdapterScrollListener) {
40             onAdapterScrollListener.onScrollStateChanged(view, scrollState);
41         }
42 
43         // 设置是否滚动的状态
44         if (scrollState == AbsListView.OnScrollListener.SCROLL_STATE_IDLE) { // 不滚动状态
45             isScrolling = false;
46             this.notifyDataSetChanged();
47         } else {
48             isScrolling = true;
49         }
50     }
51 }
复制代码

如上述代码所示,该Adapter实现了AbsListView.OnScrollListener,并在构造方法中给ListView绑定了OnScrollListener,在实现的onScrollStateChanged方法中获取到当前滚动状态,并且保存这个状态isScrolling,并暴露isScrolling()方法给外面。

OnAdapterScrollListener这个接口是继承了AbsListView.OnScrollListener,因为这里在Adapter中一景设置了OnScrollListener了,所以如果在外面设置了新的OnScrollListener的话,就会失效了,所以必须提供另外一个setOnAdapterScrollListener,然后再传入一个OnScrollListener,然后在每个方法中进行回调就好了,因为考虑到以后可能会扩展其他的接口方法,所以这里新写了一个接口(为了以后扩展时原来的代码不会被影响,推荐使用OnAdapterScrollSampleListener这个实现类来代替OnAdapterScrollListener这个接口,OnAdapterScrollSampleListener这个类只是对OnAdapterScrollListener的所有方法进行了空实现)。

 

然后我们编写一个MyAdapter继承ABaseAdapter,然后,在getView()方法中,需要显示头像的时候调用如下方法:

// 如果在滚动(从内存中查找,找不到也不进行网络请求)

ImageLoader.getInstances().displayImage(headUrl, headIv, null, R.drawable.default_head, isScrolling());

看到木有?

1. displayImage()方法发生了改变,多了最后一个参数isOnlyMemory这个参数,表示是否只是在内存缓存中找这张图片,如果没有就不再继续找下去了(displayImage原来的方法我还留着,所以不会影响之前的代码)。

2. 调用了isScrolling()方法,作为参数isOnlyMemory的值,表示,如果正在滚动的话,就只在缓存中找这张图片。

这样,运行原来的代码试试吧,是不是效率快了一些?

android listView 重写Adapter方法getView的性能优化问题

1.为什么要重写Adapter 的getView方法      为什么要重写getView方法 可以看这里   android listView滚动时checkBox的状态不会保存的问题    2.在...
  • dxswzj
  • dxswzj
  • 2013年08月31日 00:47
  • 10373

【Win 10应用开发】延迟加载图片的另一种方法

上一篇文章中老周给大伙介绍了x:Phase和x:Bind的用法,并演示了一个延迟加载的示例。不过,那个例子会遗留一个问题,就是UI线程被阻塞,所以启动应用较慢。 如果希望图片可以延迟加载,或许我...
  • bfboys
  • bfboys
  • 2016年09月27日 15:14
  • 125

jquery插件lazyload.js延迟加载图片的使用方法

来源:http://www.jb51.net/article/46985.htm 如果一个网页很长并且有很多图片的话,下载图片就需要很多时间,那么就会影响整个网页的加载速度,而这款延迟加载插件...

html延迟加载图片

  • 2016年03月08日 08:54
  • 1KB
  • 下载

DIY的JS延迟加载图片的插件

  • 2011年06月18日 17:21
  • 67KB
  • 下载

通用Adapter与ListView滚动时不加载图片的封装

摘要 本文原创,转载请注明地址:http://kymjs.com/code/2015/04/28/01 介绍一个通用的Adapter实现,和对ListView滚动时不加载图片的封装。 在...

Android ListView优化 异步加载图片

  • 2015年07月20日 09:59
  • 5.19MB
  • 下载

滚动条滚动延迟加载图片

  • 2012年12月09日 13:47
  • 929KB
  • 下载

Android的ListView异步加载图片时,错位、重复、闪烁问题的分析及解决方法

Android ListView异步加载图片错位、重复、闪烁分析以及解决方案,具体问题分析以及解决方案请看下文。我们在使用ListView异步加载图片的时候,在快速滑动或者网络不好的情况下,会出现图片...

jquery实现真正的延迟加载图片

  • 2011年11月03日 14:39
  • 896B
  • 下载
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:[Android]ListView的Adapter.getView()方法中延迟加载图片的优化
举报原因:
原因补充:

(最多只允许输入30个字)