转发请标明转载地址:http://blog.youkuaiyun.com/coder_nice/article/details/54908216
引言
ListView是Android系统中最常用也是最复杂的原生控件,了解ListView的工作原理非常必要。
ListView加载非常多数据的时候,上下滑动也不会造成oom等问题,随着手指的滑动,屏幕中会移除和新增View,无缝衔接这个过程,并且不会增加更多的内存消耗,复用是listview最核心的原理。
在研究ListView源码的时候,郭神的文章ListView工作原理给我很大的帮助,在此感谢郭神。
正文
想了解ListView的工作原理,一定需要了解AdapterView,AbsListView,Adapter,ListView,RecycleBin五个部分。
可以看得到AdapterView是ViewGroup以下的最高一级,然后是AbsListView,最后才是ListView。
Adapter中文解释为适配器,生活中你一定经常听到电源适配器,通过电源适配器就很直观的理解适配器就是把让原本不能相连的两端通过适配器相连起来。
而ListView想和数据源链接起来,就需要Adapter充当适配器的角色。
RecycleBin是AbsListView的内部类,也是复用的控制中心,正是RecycleBin的存在让ListView有了复用的能力。
AdapterView
继承关系
从AdapterView继承关系图可以看的出,AdapterView是所有使用适配器的View统一的父类。
作用
如果你想使用Adapter来做数据与View之间的桥梁,那View一定是AdapterView的子类。
所以AdapterView就是专门为了Adapter适配器,基于ViewGroup封装的一个抽象类。
具体实现
AdapterView中根据Adapter,抽象出了一些共用的属性和方法。
- setAdapter()方法就是一个抽象方法,因为不同子类中setAdapter()方法中实现和逻辑有很多差别,在AbsListView子类中,setAdapter()接收的都是ListAdapter,而在 AbsSpinner的子类中接收的对象都是SpinnerAdapter,并且不同的子类中setAdapter()也相差很多,所以setAdapter()做成抽象的方法就很好理解了。
- AdapterView中对addView和removeView做了抛出异常的处理,因为AdapterView的子类中,child view的增加和减少全部依赖于数据源,随着数据源的变化而变化,不能直接调用addView和removeView来改变child view的数量。
- AdapterView中记录了child view的数量和当前屏幕显示的第一个和最后一个view的位置,实现了OnItemClickListener等监听接口。
- 还有一些关于选中的属性和方法,就不再过多解释,AdapterView类本身代码不多,而且比较简洁,大家可以自行阅读源码。
AbsListView
继承关系
从继承关系可以看出,AbsListView是所有列表View的统一父类,常见的子类就是GridView和ListView。
作用
AdapterView抽象封装的是所有使用适配器的view共用的属性和方法,AdapterView有三种使用情景,分别对应三个直接的子类,AbsListView, AbsSpinner, AdapterViewAnimator。
AbsSpinner针对的是从多个选项中选择一个,AdapterViewAnimator针对的是多个view之间切换时自定义动画,而AbsListView专门针对列表展示数据内容的这种使用情景,又进一步做了封装和抽象。
实现
GridView和ListView这种有多个item,可复用的列表类View,最核心的逻辑都是在AbsListView中实现的,GridView和ListView其实从原理上看是基本一致的,只是布局上一个是列表,一个是网格。复用、滑动、处理数据变化等最重要最核心的,也是GridView和ListView共用的逻辑部分,都在AbsListView中实现的。
可以看的出AbsListView内部结构和实现的逻辑是比较复杂的,本篇着重讲关乎工作原理的几个方法,其他的如点击事件、选中状态等方法就不再做解释,感兴趣的同学可以自行查看阅读源码。
onLayout()方法
protected void onLayout(boolean changed, int l, int t, int r, int b) {
super.onLayout(changed, l, t, r, b);
mInLayout = true;
final int childCount = getChildCount();
if (changed) {
for (int i = 0; i < childCount; i++) {