首頁 > 軟體

Android開發RecyclerView單獨重新整理使用技巧

2022-09-26 14:06:04

單刷RecycleView的子Item

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

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佈局如下,當前只想重新整理頂部內容

1.普通payload方式

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

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

mAdapter.notifyItemChanged(0, 1)

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

override fun onBindViewHolder(
    holder: RecyclerView.ViewHolder,
    position: Int,
    payloads: MutableList<Any>
) {
    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中區域性內容重新整理的關鍵。

private class InnerAdapter2 : ListAdapter<String, RecyclerView.ViewHolder>(object : DiffUtil.ItemCallback<String>() {
        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的頂部、中部還是底部。

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的某一部分內容如何實現單獨重新整理,沒什麼難度,大家知道並學會運用即可,更多關於Android RecyclerView單獨重新整理的資料請關注it145.com其它相關文章!


IT145.com E-mail:sddin#qq.com