Android控件RecyclerView與ListView的異同


在我的一篇介紹Android新控件RecyclerView的博客(Android L新控件RecyclerView簡介)中,一個讀者留言說RecyclerView跟ListView之間好像沒有什么不同,我覺得這是一個好問題,應該明確地區分一下兩者的睯,所以我就研究了一下它倆之間的區別,然后也對兩者的使用有了更加深入的了解。

Android是一個不斷進化的平台,Android 5.0的v7版本支持包中引入了新的RecyclerView控件,正如官方文檔所言,RecyclerView是ListView的豪華增強版。它主要包含以下幾處新的特性,如ViewHolder,ItemDecorator,LayoutManager,SmothScroller以及增加或刪除item時item動畫等。官方推薦我們采用RecyclerView來取代ListView。

ViewHolder
ViewHolder是用來保存視圖引用的類,無論是ListView亦或是RecyclerView。只不過在ListView中,ViewHolder需要自己來定義,且這只是一種推薦的使用方式,不使用當然也可以,這不是必須的。只不過不使用ViewHolder的話,ListView每次getView的時候都會調用findViewById(int),這將導致ListView性能展示遲緩。而在RecyclerView中使用RecyclerView.ViewHolder則變成了必須,盡管實現起來稍顯復雜,但它卻解決了ListView面臨的上述不使用自定義ViewHolder時所面臨的問題。RecyclerView.ViewHolder被BaseAdapter使用,以將posiiton綁定到上面(可以通過API查看RecyclerView.ViewHolder#getPosition()方法)。

LayoutManager
我們知道ListView只能在垂直方向上滾動,Android API沒有提供ListView在水平方向上面滾動的支持。或許有多種方式實現水平滑動,但是請想念我,ListView並不是設計來做這件事情的。但是RecyclerView相較於ListView,在滾動上面的功能擴展了許多。它可以支持多種類型列表的展示要求,主要如下:

LinearLayoutManager,可以支持水平和豎直方向上滾動的列表。

StaggeredGridLayoutManager,可以支持交叉網格風格的列表,類似於瀑布流或者Pinterest。

GridLayoutManager,支持網格展示,可以水平或者豎直滾動,如展示圖片的畫廊。


ItemAnimator
列表動畫是一個全新的、擁有無限可能的維度。起初的Android API中,刪除或添加item時,item是無法產生動畫效果的。后面隨着Android的進化,Google的Chat Hasse推薦使用ViewPropertyAnimator屬性動畫來實現上述需求。
相比較於ListView,RecyclerView.ItemAnimator則被提供用於在RecyclerView添加、刪除或移動item時處理動畫效果。同時,如果你比較懶,不想自定義ItemAnimator,你還可以使用DefaultItemAnimator

Adapter
ListView的Adapter中,getView是最重要的方法,它將視圖跟position綁定起來,是所有神奇的事情發生的地方。同時我們也能夠通過registerDataObserver在Adapter中注冊一個觀察者。RecyclerView也有這個特性,RecyclerView.AdapterDataObserver就是這個觀察者。ListView有三個Adapter的默認實現,分別是ArrayAdapter、CursorAdapter和SimpleCursorAdapter。然而,RecyclerView的Adapter則擁有除了內置的內DB游標和ArrayList的支持之外的所有功能。RecyclerView.Adapter的實現的,我們必須采取措施將數據提供給Adapter,正如BaseAdapter對ListView所做的那樣。

ItemDecoration
在ListView中如果我們想要在item之間添加間隔符,我們只需要在布局文件中對ListView添加如下屬性即可:

1 android:divider="@android:color/transparent"
2 android:dividerHeight="5dp"
View Code

有趣的是,RecyclerView在默認情況下並不在item之間展示間隔符。盡管Google的家伙有意地將這個問題遺留給我們去自定義間隔符,但這的確增加了開發人員的負擔。如果你想要添加間隔符,你必須使用RecyclerView.ItemDecoration類來實現。或者,你可以應用官方示例中的DividerItemDecoration.java文件。

OnItemTouchListener
ListView通過AdapterView.OnItemClickListener接口來探測點擊事件。而RecyclerView則通過RecyclerView.OnItemTouchListener接口來探測觸摸事件。它雖然增加了實現的難度,但是卻給予開發人員攔截觸摸事件更多的控制權限。

Others
ListView可以設置選擇模式,並添加MultiChoiceModeListener,如下所示:

 1 listView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE_MODAL);
 2 listView.setMultiChoiceModeListener(new MultiChoiceModeListener() {
 3     public boolean onCreateActionMode(ActionMode mode, Menu menu) { ... }
 4     public void onItemCheckedStateChanged(ActionMode mode, int position,
 5 long id, boolean checked) { ... }
 6     public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
 7         switch (item.getItemId()) {
 8             case R.id.menu_item_delete_crime:
 9             CrimeAdapter adapter = (CrimeAdapter)getListAdapter();
10             CrimeLab crimeLab = CrimeLab.get(getActivity());
11             for (int i = adapter.getCount() - 1; i >= 0; i--) {
12                 if (getListView().isItemChecked(i)) {
13                     crimeLab.deleteCrime(adapter.getItem(i));
14                 }
15           }
16         mode.finish();
17         adapter.notifyDataSetChanged();
18         return true;
19         default:
20             return false;
21 }
22     public boolean onPrepareActionMode(ActionMode mode, Menu menu) { ... }
23     public void onDestroyActionMode(ActionMode mode) { ... }
24 });
View Code

而RecyclerView則沒有此功能。

總之,通過比較我們可以發現,RecyclerView充滿了大量的自定義功能,它可以用於實現復雜的列表或網格,但實現起來稍顯得復雜。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM