RecyclerView是ListView的增强版。有了它之后,你就可以抛弃ListView了。
recycle,重复利用。在ListView里,我们得自己写重复利用View的代码,而RecyclerView则是将此内置进去,并提供清晰的结构。
设计模式相关
RecyclerView 和 ListView 都是使用对象适配器模式。我专门为此写了一篇对适配器模式的介绍:设计模式:适配器模式
以下是该模式的UML类图(方便起见,为箭头加了说明):
对应到 RecyclerView :
类图 | RecyclerView |
---|---|
Client | android.support.v7.widget.RecyclerView |
Target | android.support.v7.widget.RecyclerView.Adapter |
Adapter | 自己继承RecyclerView.Adapter并实现 |
Adaptee | 需要展示的List数据 |
这里RecyclerView的Target没有再提出接口,直接写成抽象类
远程依赖库
如果使用RecyclerView,则需要在build.gradle的dependencies里添加远程依赖库:
compile 'com.android.support:recyclerview-v7:25.1.0'
这是为了让低版本的Android系统可以用到这个RecyclerView。如果不这样,则低版本无法使用。
RecyclerView 和 ListView 有什么不同?
- 继承结构不同:
- 它自带ViewHolder。我们再也不必自己添加各种判断了。
- 行布局文件不再由外部传入,而是在内部的onCreateViewHolder()里直接使用。
- 添加了横向的支持,即左右滚动。这个可以用在Android TV上。
- ……
RecyclerView的Adapter(以下称为RecyclerViewAdapter)内部需要重写的几个部分:
-
通过构造方法获得List
ListViewAdapter也是通过构造方法获得List,但还有另外两个参数,分别是Context和行布局的resourceId。
-
通过onCreateViewHolder()加载布局
ListViewAdapter在getView()里加载。
-
通过创建内部类并继承RecyclerView.ViewHolder来获取布局里的控件
ListViewAdapter也是创建内部类,但它是把findViewById()放到getView()里。RecyclerViewAdapter则是放在这个内部类里面。
-
通过onBindViewHolder()给控件赋值
Bind即绑定,意思是把数据绑定到ViewHolder上。
ListViewAdapter把这个动作放在getView()里。
从这几个部分可以看到,本来getView()里做的三个动作,现在分到三个不同的地方。
如下图,左边是ListView的Adapter,右边是RecyclerView的Adapter:
除此之外,RecyclerView需要在初始化它的地方给它传入LayoutManager。
如下图,左边是ListView的Activity,右边是RecyclerView的Activity:
ListView 自带三种Adapter的实现。例如很常用的ArrayAdapter,而RecyclerView则没有这样的实现,必须自己写Adapter并重写上面所说的各个方法。
虽然RecyclerView的Adapter没有ListView的三种方便的实现,但它有三个子类:
- ItemBridgeAdapter
- LeanbackListPreferenceDialogFragment.AdapterMulti
- LeanbackListPreferenceDialogFragment.AdapterSingle
这三个是给媒体设备(电视)或者游戏用的。所以如果你想要ArrayAdapter
;SimpleAdapter
;SimpleCursorAdapter
的效果,你得通过继承RecyclerView.Adapter来自己实现。
这里有个RecyclerView对SimpleCursorAdapter
的实现:CursorRecyclerAdapter.java
效果
左边是ListView,右边是RecyclerView:
RecyclerView默认没有分割线。
如果想要分割线,可以使用默认的DividerItemDecoration:
DividerItemDecoration divider =
new DividerItemDecoration(recyclerView.getContext(),
layoutManager.getOrientation());
recyclerView.addItemDecoration(divider);
也可以通过继承RecyclerView.ItemDecoration来自定义。
点击事件
由于每一个Item里面可能有多个视图,如果想直接点击这些视图,在ListView里实现比较麻烦。
与ListView的setOnItemClickListener()不同,RecyclerView需要在其Adapter的onCreateViewHolder()里设置点击事件。