在搞懂這個(gè)問(wèn)題之前,我們最好先搞明白ListView和RecyclerView的實(shí)現(xiàn)原理,這里推薦兩篇文章:
那么,ListView和RecyclerView的主要區(qū)別有哪些呢?
一、緩存機(jī)制的不同
這里可以參考《Android ListView 與 RecyclerView 對(duì)比淺析—緩存機(jī)制》
二馆纳、布局效果鲸湃、常用功能與API等
參考《ListView 與 RecyclerView 簡(jiǎn)單對(duì)比》
使用上
ListView:
- 繼承重寫(xiě)B(tài)aseAdapter類(lèi);
- 自定義ViewHolder與convertView的優(yōu)化;
RecyclerView:
- 繼承重寫(xiě)RecyclerView.Adapter與RecyclerView.ViewHolder
- 設(shè)置LayoutManager挟鸠,以及l(fā)ayout的布局效果
區(qū)別:
- ViewHolder的編寫(xiě)規(guī)范化童太,ListView是需要自己定義的,而RecyclerView是規(guī)范好的栓始;
- RecyclerView復(fù)用item全部搞定务冕,不需要像ListView那樣setTag()與getTag();
- RecyclerView多了一些LayoutManager工作幻赚,但實(shí)現(xiàn)了布局效果多樣化禀忆;
布局效果
- ListView 的布局比較單一,只有一個(gè)縱向效果坯屿;
- RecyclerView 的布局效果豐富油湖, 可以在LayoutMananger中設(shè)置:線(xiàn)性布局(縱向,橫向)领跛,表格布局乏德,瀑布流布局
- 在RecyclerView 中,如果存在的LayoutManager不能滿(mǎn)足需求吠昭,可以在LayoutManager的API中自定義Layout:
例如:scrollToPosition(), setOrientation(), getOrientation(), findViewByPosition()等等喊括;
空數(shù)據(jù)處理
在ListView中有個(gè)setEmptyView() 用來(lái)處理Adapter中數(shù)據(jù)為空的情況;但是在RecyclerView中沒(méi)有這個(gè)API矢棚,所以在RecyclerView中需要進(jìn)行一些數(shù)據(jù)判斷來(lái)實(shí)現(xiàn)數(shù)據(jù)為空的情況郑什;
HeaderView 與 FooterView
- 在ListView中可以通過(guò)addHeaderView() 與 addFooterView()來(lái)添加頭部item與底部item,來(lái)當(dāng)我們需要實(shí)現(xiàn)下拉刷新或者上拉加載的情況蒲肋;而且這兩個(gè)API不會(huì)影響Adapter的編寫(xiě)蘑拯;
- 但是RecyclerView中并沒(méi)有這兩個(gè)API钝满,所以當(dāng)我們需要在RecyclerView添加頭部item或者底部item的時(shí)候,我們可以在A(yíng)dapter中自己編寫(xiě)申窘,根據(jù)ViewHolder的Type與View來(lái)實(shí)現(xiàn)自己的Header弯蚜,F(xiàn)ootter與普通的item,但是這樣就會(huì)影響到Adapter的數(shù)據(jù)剃法,比如position碎捺,添加了Header與Footter后,實(shí)際的position將大于數(shù)據(jù)的position贷洲;
局部刷新
- 在ListView中通常刷新數(shù)據(jù)是用notifyDataSetChanged() 收厨,但是這種刷新數(shù)據(jù)是全局刷新的(每個(gè)item的數(shù)據(jù)都會(huì)重新加載一遍),這樣一來(lái)就會(huì)非常消耗資源优构;
- RecyclerView中可以實(shí)現(xiàn)局部刷新诵叁,例如:notifyItemChanged();
- 但是如果要在ListView實(shí)現(xiàn)局部刷新俩块,依然是可以實(shí)現(xiàn)的黎休,當(dāng)一個(gè)item數(shù)據(jù)刷新時(shí),我們可以在A(yíng)dapter中玉凯,實(shí)現(xiàn)一個(gè)onItemChanged()方法势腮,在方法里面獲取到這個(gè)item的position(可以通過(guò)getFirstVisiblePosition()),然后調(diào)用getView()方法來(lái)刷新這個(gè)item的數(shù)據(jù)漫仆;
動(dòng)畫(huà)效果:
- 在RecyclerView中捎拯,已經(jīng)封裝好API來(lái)實(shí)現(xiàn)自己的動(dòng)畫(huà)效果;有許多動(dòng)畫(huà)API盲厌,例如:notifyItemChanged(), notifyDataInserted(), notifyItemMoved()等等署照;如果我們需要淑賢自己的動(dòng)畫(huà)效果,我們可以通過(guò)相應(yīng)的接口實(shí)現(xiàn)自定義的動(dòng)畫(huà)效果(RecyclerView.ItemAnimator類(lèi))吗浩,然后調(diào)用RecyclerView.setItemAnimator() (默認(rèn)的有SimpleItemAnimator與DefaultItemAnimator)建芙;
- 但是ListView并沒(méi)有實(shí)現(xiàn)動(dòng)畫(huà)效果,但我們可以在A(yíng)dapter自己實(shí)現(xiàn)item的動(dòng)畫(huà)效果懂扼;
ItemTouchHelper:
- 創(chuàng)建ItemTouchHelper實(shí)例禁荸,然后在ItemTouchHelper.SimpleCallback(),然后在Callback中實(shí)現(xiàn)getMovementFlags(), onMove(), onSwiped()阀湿, 最后調(diào)用RecyclerView的attachToRecyclerView方法赶熟;
Item點(diǎn)擊事件:
- 在ListView中有onItemClickListener(), onItemLongClickListener(), onItemSelectedListener(), 但是添加HeaderView與FooterView后就不一樣了,因?yàn)镠eaderView與FooterView都會(huì)算進(jìn)position中陷嘴,這時(shí)會(huì)發(fā)現(xiàn)position會(huì)出現(xiàn)變化映砖,可能會(huì)拋出數(shù)組越界,為了解決這個(gè)問(wèn)題灾挨,我們?cè)趃etItemId()方法(在該方法中HeaderView與FooterView返回的值是-1)中通過(guò)返回id來(lái)標(biāo)志對(duì)應(yīng)的item邑退,而不是通過(guò)position來(lái)標(biāo)記竹宋;但是我們可以在A(yíng)dapter中針對(duì)每個(gè)item寫(xiě)在getView()中會(huì)比較合適;
- 而在RecyclerView中地技,提供了唯一一個(gè)API:addOnItemTouchListener()逝撬,監(jiān)聽(tīng)item的觸摸事件;我們可以通過(guò)RecyclerView的addOnItemTouchListener()加上系統(tǒng)提供的Gesture Detector來(lái)實(shí)現(xiàn)像ListView那樣監(jiān)聽(tīng)某個(gè)item某個(gè)操作方法乓土;
嵌套滾動(dòng)機(jī)制:
- 在事件分發(fā)機(jī)制中,Touch事件在進(jìn)行分發(fā)的時(shí)候溯警,由父View向子View傳遞趣苏,一旦子View消費(fèi)這個(gè)事件的話(huà),那么接下來(lái)的事件分發(fā)的時(shí)候梯轻,父View將不接受食磕,由子View進(jìn)行處理;但是與Android的事件分發(fā)機(jī)制不同喳挑,嵌套滾動(dòng)機(jī)制(Nested Scrolling)可以彌補(bǔ)這個(gè)不足彬伦,能讓子View與父View同時(shí)處理這個(gè)Touch事件,主要實(shí)現(xiàn)在于NestedScrollingChild與NestedScrollingParent這兩個(gè)接口伊诵;而在RecyclerView中单绑,實(shí)現(xiàn)的是NestedScrollingChild,所以能實(shí)現(xiàn)嵌套滾動(dòng)機(jī)制曹宴;
- ListView就沒(méi)有實(shí)現(xiàn)嵌套滾動(dòng)機(jī)制搂橙;