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


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

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的值,表示,如果正在滚动的话,就只在缓存中找这张图片。

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

 



来源博客:Wang Jie's Blog
本文链接:https://blog.wangjiegulu.com/2014/12/03/Android-ListView的Adapter-getView-方法中延迟加载图片的优化/
版权声明:本博客所有文章除特别声明外,均采用 CC BY 4.0 CN协议 许可协议。转载请注明出处。