數見不鮮的RecyclerView使用技巧,你瞭解嗎(一)?

語言: CN / TW / HK

highlight: vs theme: devui-blue


我報名參加金石計劃1期挑戰——瓜分10萬獎池,這是我的第1篇文章,點選檢視活動詳情

本篇文章主要是分享下RecyclerView中子item如何單獨重新整理以及子item的某一部分內容如何實現單獨重新整理,希望可以給你帶來幫助。

單刷RecycleView的子Item

除非必要,應該儘可能避免呼叫notifyDataSetChanged()去重新整理RecyclerView列表 ,這會對效能造成影響,所以RecyclerView.Adapter還提供了一下幾個方法:

kotlin private fun recycleViewRelated() { mAdapter.notifyItemChanged() mAdapter.notifyItemRangeChanged() mAdapter.notifyItemInserted() mAdapter.notifyItemRangeInserted() mAdapter.notifyItemRangeRemoved() mAdapter.notifyItemRemoved() }

  • notifyItemXXXChanged():通知指定索引的item呼叫onBindViewHolder()重新整理介面

  • notifyItemXXXInserted():通知在指定索引處插入item,即插入ViewHolder,有可能複用,有可能重新呼叫onCreateViewHolder()建立

  • notifyItemXXXRemoved(): 通知移除指定索引的item,即移除ViewHolder,並根據情況放到大家瞭解的1級mAttachedScrap快取或3級快取RecycledViewPool

上面的幾個情況大家根據情況呼叫,請注意呼叫上面單刷方法時,同樣要保證RecyclerView資料來源也進行了插入/刪除/更新操作,否則可能會引發不必要的異常。

單刷RecycleView的子Item的區域性內容

有時候,某個子Item對應的佈局比較複雜,且每次重新整理只可能會重新整理其中一部分,這個時候使用上面介紹的單刷就會重新整理到子item中不必要重新整理的部分,這個時候我們可以藉助payload實現單刷Item中的某一部分內容。

接下來介紹兩種方式,假設當前Item佈局如下,當前只想重新整理頂部內容

image.png

1.普通payload方式

這個主要是藉助帶payload引數的notifyItemXXXRemoved/Changed/Inserted()方法+帶payloads引數的onBindViewHolder()方法實現,接下來我們來看程式碼實操。

我們用1、2、3分別來標識Item的頂部、中部和底部對應佈局內容,目前我們只想重新整理標識為1對應的頂部內容:

kotlin mAdapter.notifyItemChanged(0, 1) 上面程式碼就代表著重新整理下標為0對應item佈局的頂部內容,接下來我們在onBindViewHolder() 中處理:

``` override fun onBindViewHolder( holder: RecyclerView.ViewHolder, position: Int, payloads: MutableList ) { if (payloads.isEmpty()) { super.onBindViewHolder(holder, position, payloads) return }

when(payloads[0]) {
    //重新整理頂部內容
    1 -> {}
    //重新整理中部內容
    2 -> {}
    //重新整理底部內容
    3 -> {}
}

} ```

得用帶有如上payloads引數的onBindViewHolder()才能處理通過notifyItemChanged()最後一個引數傳遞過來的payload引數。

請注意當payloads集合引數為空時,要主動呼叫super.onBindViewHolder(holder, position, payloads)保證單item的整體重新整理。

2.ListAdapter方式

ListAdapter是基於DiffUtil實現列表中部分item重新整理的,具體的使用這裡不做過多介紹。

當我們繼承ListAdapter自定義一個介面卡時,要傳入一個DiffUtil.ItemCallback物件,這個物件有個getChangePayload()方法,這裡就是實現item中區域性內容重新整理的關鍵。

```kotlin private class InnerAdapter2 : ListAdapter(object : DiffUtil.ItemCallback() { override fun areItemsTheSame(oldItem: String, newItem: String): Boolean { }

    override fun areContentsTheSame(oldItem: String, newItem: String): Boolean {
    }

    override fun getChangePayload(oldItem: String, newItem: String): Any? {
        return super.getChangePayload(oldItem, newItem)
    }
})

```

我們只需要對上面的getChangePayload()方法重寫,根據資料變更的範圍來決定重新整理item的頂部、中部還是底部。
```kotlin override fun getChangePayload(oldItem: String, newItem: String): Any? {

val list = mu

//如果資料變更會影響item頂部內容顯示,則返回1重新整理item頂部內容
if (oldItem != newItem) {
    return 1
}

//如果資料變更會影響item頂部內容顯示,則返回2重新整理item中部內容
if (oldItem != newItem) {
    return 2
}

//如果資料變更會影響item頂部內容顯示,則返回3重新整理item底部內容
if (oldItem != newItem) {
    return 3
}

return super.getChangePayload(oldItem, newItem)

} ```

接下來我們在onBindViewHolder中處理就行 ,處理的方式和上面相同,這裡就再描述。

如果我們想要同時實現item中頂部和底部佈局內容的同時重新整理,那就可以向onBindViewHolder()的payload中傳一個集合包含1和3標識或者其他特殊標識等等,方式不限,只要能讓onBindViewHolder()知道要重新整理頂部和底部就即可。

總結

本篇文章主要是講解了如何實現RecyclerView中子item如何單獨重新整理以及子item的某一部分內容如何實現單獨重新整理,沒什麼難度,大家知道並學會運用即可 。