Android学习笔记之ListView复用机制详解

PS:满打满算,差不多三个月没写博客了...前一阵忙的不可开交...总算是可以抽出时间研究研究其他事情了...

1.ListView的复用机制

ListView是我们经常使用的一个控件,虽然说都会用,但是却并不一定完全清楚ListView的复用机制,虽然在Android 5.0版本之后提供了RecycleView去替代ListView和GridView,提供了一种插拔式的体验,也就是所谓的模块化。本篇主要针对ListView的复用机制进行探讨,因此就 提RecycleView。昨天看了一下郭霖大神的ListView原理深度解析的一篇博客,因此学习了一段时间,自己也说一下自己的理解。

i.RecycleBin的基本原理

首先需要说一下RecycleBin的基本原理,这个类也是实现复用的关键类。接着我们需要明确ActiveView的概念,ActivityView其实就是在UI屏幕上可见的视图(onScreenView),也是与用户进行交互的View,那么这些View会通过RecycleBin直接存储到mActivityView数组当中,以便为了直接复用,那么当我们滑动ListView的时候,有些View被滑动到屏幕之外(offScreen) View,那么这些View就成为了ScrapView,也就是废弃的View,已经无法与用户进行交互了,这样在UI视图改变的时候就没有绘制这些无用视图的必要了。他将会被RecycleBin存储到mScrapView数组当中,但是没有被销毁掉,目的是为了二次复用,也就是间接复用。当新的View需要显示的时候,先判断mActivityView中是否存在,如果存在那么我们就可以从mActivityView数组当中直接取出复用,也就是直接复用,否则的话从mScrapView数组当中进行判断,如果存在,那么二次复用当前的视图,如果不存在,那么就需要inflate View了。

这是一个总体的流程图,复用机制就是这样的。那么我们先来理解一下ListView第一次加载的时候都做了哪些工作,首先会执行onLayout方法。。

/**
 * Subclasses should NOT override this method but {@link #layoutChildren()}
 * instead.
 */
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
  super.onLayout(changed, l, t, r, b);
  mInLayout = true;
  if (changed) {
    int childCount = getChildCount();
    for (int i = 0; i < childCount; i++) {
      getChildAt(i).forceLayout();
    }
    mRecycler.markChildrenDirty();
  }
  layoutChildren();
  mInLayout = false;
}

这里可以看到onLayout方法会调用layoutChildren()方法,也就是对item进行布局的流程,layoutChildren()方法就不进行粘贴了,代码量过长我们只需要知道,这是对ListView中的子View进行布局的一个方式就可以了,在我们第一次加载ListView的时候,RecycleBin中的数组都没有任何的数据,因此第一次加载都需要inflate View,也就是创建新的View。并且第一次加载的时候是自顶向下对数据进行加载的,因此在layoutChildren()会执行fillFromTop()方法。fillFromTop()会执行filleDown()方法。

/**
 * Fills the list from pos down to the end of the list view.
 *
 * @param pos The first position to put in the list
 *
 * @param nextTop The location where the top of the item associated with pos
 *    should be drawn
 *
 * @return The view that is currently selected, if it happens to be in the
 *     range that we draw.
 *
 * @param pos:列表中的一个绘制的Item在Adapter数据源中对应的位置
 * @param nextTop:表示当前绘制的Item在ListView中的实际位置..
 */
private View fillDown(int pos, int nextTop) {
  View selectedView = null;
  /**
   * end用来判断Item是否已经将ListView填充满
   */
  int end = (getBottom() - getTop()) - mListPadding.bottom;
  while (nextTop < end && pos < mItemCount) {
     /**
     * nextTop < end确保了我们只要将新增的子View能够覆盖ListView的界面就可以了
     *pos < mItemCount确保了我们新增的子View在Adapter中都有对应的数据源item
     */
    // is this the selected item?
    boolean selected = pos == mSelectedPosition;
    View child = makeAndAddView(pos, nextTop, true, mListPadding.left, selected);
    /**
      *将最新child的bottom值作为下一个child的top值,存储在nextTop中
      */
    nextTop = child.getBottom() + mDividerHeight;
    if (selected) {
      selectedView = child;
    }
    pos++;
  }
  return selectedView;
}
  • 在while循环中添加子View,我们先不看while循环的具体条件,先看一下循环体。在循环体中,将pos和nextTop传递给makeAndAddView方法,该方法返回一个View作为child,该方法会创建View,并把该View作为child添加到ListView的children数组中。
  • 然后执行nextTop = child.getBottom() + mDividerHeight,child的bottom值表示的是该child的底部到ListView顶部的距离,将该child的bottom作为下一个child的top,也就是说nextTop一直保存着下一个child的top值。
  • 最后调用pos++实现position指针下移。现在我们回过头来看一下while循环的条件while (nextTop < end && pos < mItemCount)。
  • nextTop < end确保了我们只要将新增的子View能够覆盖ListView的界面就可以了,比如ListView的高度最多显示10个子View,我们没必要向ListView中加入11个子View。
  • pos < mItemCount确保了我们新增的子View在Adapter中都有对应的数据源item,比如ListView的高度最多显示10个子View,但是我们Adapter中一共才有5条数据,这种情况下只能向ListView中加入5个子View,从而不能填充满ListView的全部高度。

这里存在一个关键方法,也就是makeAndAddView()方法,这是ListView将Item显示出来的核心部分,也是这个部分涉及到了ListView的复用

private View makeAndAddView(int position, int y, boolean flow, int childrenLeft,
    boolean selected) {
  View child;
  //判断数据源是否发生了变化.
  if (!mDataChanged) {
    // Try to use an exsiting view for this position
    //如果mActivityView[]数组中存在可以直接复用的View,那么直接获取,然后重新布局.
    child = mRecycler.getActiveView(position);
    if (child != null) {
      // Found it -- we're using an existing child
      // This just needs to be positioned
      setupChild(child, position, y, flow, childrenLeft, selected, true);
      return child;
    }
  }
  // Make a new view for this position, or convert an unused view if possible
  /**
   *如果mActivityView[]数组中没有可用的View,那么尝试从mScrapView数组中读取.然后重新布局.
   *如果可以从mScrapView数组中可以获取到,那么直接返回调用mAdapter.getView(position,scrapView,this);
   *如果获取不到那么执行mAdapter.getView(position,null,this)方法.
   */
  child = obtainView(position, mIsScrap);
  // This needs to be positioned and measured
  setupChild(child, position, y, flow, childrenLeft, selected, mIsScrap[0]);
  return child;
}

这里可以看到如果数据源没有变化的时候,会从mActivityView数组中判断是否存在可以直接复用的View,可能很多读者都不太明白直接复用到底是怎么个过程,举个例子,比如说我们ListView一页可以显示10条数据,那么我们在这个时候滑动一个Item的距离,也就是说把position = 0的Item移除屏幕,将position = 10 的Item移入屏幕,那么position = 1的Item是不是就直接能够从mActivityView数组中拿到呢?这是可以的,我们在第一次加载Item数据的时候,已经将position = 0~9的Item加入到了mActivityView数组当中,那么在第二次加载的时候,由于position = 1 的Item还是ActivityView,那么这里就可以直接从数组中获取,然后重新布局。这里也就表示的是Item的直接复用。

如果我们在mActivityView数组中获取不到position对应的View,那么就尝试从mScrapView废弃View数组中尝试去获取,还拿刚才的例子来说当position = 0的Item被移除屏幕的时候,首先会Detach让View和视图进行分离,清空children,然后将废弃View添加到mScrapView数组当中,当加载position = 10的Item时,mActivityView数组肯定是没有的,也就无法获取到,同样mScrapView中也是不存在postion = 10与之对应的废弃View,说白了就是mScrapView数组只有mScrapView[0]这一项数据,肯定是没有mScrapView[10]这项数据的,那么我们就会这样想,肯定是从Adapter中的getView方法获取新的数据喽,其实并不是这样,虽然mScrapView中虽然没有与之对应的废弃View,但是会返回最后一个缓存的View传递给convertview。那么也就是将mScrapView[0]对应的View返回。总体的流程就是这样。

这里我们可以看到,ListView始终只会在getView方法中inflate一页的Item,也就是new View只会执行一页Item的次数。后续的Item通过直接复用和间接复用完成。

注意一种情况:比如说还是一页的Item,但是position = 0的Item没有完全滑动出UI,position = 10的Item没有完全进入到UI的时候,那么position = 0的Item不会被detach掉,同样不会被加入到废弃View数组,这时mScrapView是空的,没有任何数据,那么position = 10的Item即无法从mActivityView中直接复用View,因为是第一次加载。mActivityView[10]是不存在的,同时mScrapView是空的,因此position = 10的Item只能重新生成View,也就是从getView方法中inflate。这里obtainView方法没有具体贴出,大家可以自己进去看看。obtainView其实就是判断能否从废弃View中获取到View,获取到了则执行:

if (scrapView != null) {
  child = mAdapter.getView(position, scrapView, this);
} 

这里是可以获取到,那么getView会传递scrapView。否则的话:

else {
  child = mAdapter.getView(position, null, this);
} 

获取不到就传递null,这样就会执行我们定义的Adapter中的方法。

@Override
public View getView(int position, View convertView, ViewGroup parent) {
  if(convertView == null){
    convertView = View.inflate(context, R.layout.list_item_layout, null);
  }
  return convertView;
}

至于向上滑动会执行其他的一些方法,也就是自底向上铺满ListView,同样也会直接或者间接复用控件。理解了复用的机制才是关键,因此向上滑基本就不难理解了。补充一点,RecycleBin中还存在一个方法,setViewTypeCount()方法。这个是针对Adapter中的getViewTypeCount()设定的。针对每一种数据类型,setViewTypeCount()会为每种数据类型开启一个单独的RecycleBin回收机制。这里我们只需要知道就可以了。至于在郭神博客中看到ListView会onLayout多次,这是肯定的,由于Android View加载机制问题,子控件需要根据父控件的大小要重新测量大小,经过多次测量才能够显示在UI上。这是View测量多次的原因。至于ListView在多次布局的问题我就不进行赘余了,总之无论几次测量,ListView是不会多次执行重复的逻辑的,也就是说数据不会有多份,只会存在一份数据。

这里也就是ListView复用的基本原理和RecycleBin的回收机制了。代码贴的很少,都是一些关键代码,没必要去一行一行的研究代码,毕竟和大神还差很大的一个档次。我们只需要知道这个执行过程和原理就可以了。

2.ViewHolder

最后说一说ViewHolder这个东西,很多Android学习者会把这个东西和ListView的复用机制搞混。这里ViewHolder也是在复用的时候进行使用,但是和复用机制是没太大关系的。

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    final ViewHolder holder;
    ListViewItem itemData = items.get(position);
    if(convertView == null){
      convertView = View.inflate(context, R.layout.list_item_layout, null);
      holder = new ViewHolder();
      holder.userImg = (ImageView) convertView.findViewById(R.id.user_header_img);
      holder.userName = (TextView) convertView.findViewById(R.id.user_name);
      holder.userComment = (TextView) convertView.findViewById(R.id.user_coomment);
      convertView.setTag(holder);
    }else{
      holder = (ViewHolder) convertView.getTag();
    }
    holder.userImg.setImageResource(itemData.getUserImg());
    holder.userName.setText(itemData.getUserName());
    holder.userComment.setText(itemData.getUserComment());
    return convertView;
}

static class ViewHolder{
    ImageView userImg;
    TextView userName;
    TextView userComment;
}

在实现Adapter的时候,我们一般会加上ViewHolder这个东西,ViewHolder和复用机制和原理是无关的,他的主要目的是持有Item中控件的引用,从而减少findViewById()的次数,因为findViewById()方法也是会影响效率的,因此在复用的时候他起的作用是这个,减少方法执行次数增加效率。这里做个简单的提醒,别弄混就行。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android之带group指示器的ExpandableListView(自写)

    我们都知道Android缺省的ExpandableListView的group header无法固定在界面上,当向下滚动后,不能对当前显示的那些child 指示出它们归属于哪个group,在网上搜了很多关于仿手机QQ好友分组效果的ExpandableListView,发现都不尽如意,于是乎在别人的基础上改进了一点点,其实原理还是差不多的,只是增加了往上挤出去的动画效果,而且更加简单,只不过还是没有完全到达跟QQ一样的效果,希望有高手能实现更加逼真的效果,下面我们先看看效果图:  我这里没有把Ex

  • android ListView内数据的动态添加与删除实例代码

    main.xml 文件: 复制代码 代码如下: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_pa

  • android ListView和GridView拖拽移位实现代码

    关于ListView拖拽移动位置,想必大家并不陌生,比较不错的软件都用到如此功能了.如:搜狐,网易,百度等,但是相比来说还是百度的用户体验较好,不偏心了,下面看几个示例:             首先说一下:拖拽ListView的item就不应该可以任意移动,只应该在ListView所在的范围内,而网易的你看看我都可以移动到状态栏了,虽然你做了处理,但是用户体验我个人感觉不好,在看看百度的,不仅控制了移动范围,更不错的百度的移动起来会时时的换位,看起来相当的形象,所以我认为这样相当的棒.说明一点

  • android listview优化几种写法详细介绍

    这篇文章只是总结下getView里面优化视图的几种写法,就像孔乙己写茴香豆的茴字的几种写法一样,高手勿喷,勿笑,只是拿出来分享,有错误的地方欢迎大家指正,谢谢. listview Aviewthatshowsitemsinaverticallyscrollinglist. 一个显示一个垂直的滚动子项的列表视图在android开发中,使用listview的地方很多,用它来展现数据,成一个垂直的视图.使用listview是一个标准的适配器模式,用数据--,界面--xml以及适配器--adapter,

  • android中ListView多次刷新重复执行getView的解决方法

    以前倒是没有注意listview的getView会重复执行多次,这次因为布局比较复杂,所以在测试的时候去断点跟踪,发现同一条数据不断的重复执行.觉得很奇怪,于是上网搜索了一下.网上的解释基本一致,就是ListView布局时height和width都不是fill_parent,导致不断计算高度,不断刷新.或者说它的父容器没有设置成fill_parent. 可以布局太复杂的情况下,全部按照fill_parent去调整不现实.所以想了另一种方案,就是动态固定高度. 在程序运行后,固定ListView的

  • Android ListView数据绑定显示的三种解决方法

    首先,创建一个用于显示一个item的layout,名为item.xml 复制代码 代码如下: <?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout

  • Android自定义Adapter的ListView的思路及代码

    在开发中,我们经常使用到ListView这个控件.Android的API也提供了许多创建ListView适配器的快捷方式.例如ArrayAdapter.SimpleAdapter和SimpleCursorAdapter等.但你是否发现,如果采用这些系统自带的适配器,对于事件的响应只能局限在一个行单位.假设一行里面有一个按钮和一个图片控件,它们之间的响应操作是不一样的.若采用系统自带的适配器,就不能精确到每个控件的响应事件.这时,我们一般采取自定义适配器来实现这个比较精确地请求. ListView

  • android二级listview列表实现代码

    今天来实现以下大众点评客户端的横向listview二级列表,先看一下样式.  这种横向的listview二级列表在手机软件上还不太常见,但是使用过平板的都应该知道,在平板上市比较常见的.可能是因为平板屏幕比较大,而且也能展现更多的内容. 下面来看一下我的实现步骤. 首先自定义一个listview,代码如下: 复制代码 代码如下: public class MyListView extends ListView implements Runnable { private float mLastDo

  • Android ListView长按弹出菜单二种实现方式示例

    复制代码 代码如下: /** * 知识点1:ListView item:两种长按弹出菜单方式* 知识点2:ListView SimpleAdapter的使用* 知识点 3:在java代码中创建一个ListView*/ public class ListOnLongClickActivity extends Activity {         private LinearLayout myListViewlayout;         private ListView mListView;   

  • android ListView和ProgressBar(进度条控件)的使用方法

    ListView控件的使用:ListView控件里面装的是一行一行的数据,一行中可能有多列,选中一行,则该行的几列都被选中,同时可以触发一个事件,这种控件在平时还是用得很多的.使用ListView时主要是要设置一个适配器,适配器主要是用来放置一些数据.使用起来稍微有些复杂,这里用的是android自带的SimpleAdapter,形式如下:android.widget.SimpleAdapter.SimpleAdapter(Context context, List<? extends Map<

随机推荐