在搞懂這個(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è)置:線性布局(縱向丸升,橫向),表格布局牺氨,瀑布流布局
- 在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í)候揉阎,我們可以在Adapter中自己編寫(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í)捆探,我們可以在Adapter中然爆,實(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à)效果,但我們可以在Adapter自己實(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)記;但是我們可以在Adapter中針對(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ī)制丈屹;