日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學無先后,達者為師

網站首頁 編程語言 正文

Android開發RecyclerView單獨刷新使用技巧_Android

作者:長安皈故里 ? 更新時間: 2022-11-14 編程語言

單刷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()知道要刷新頂部和底部就即可。

總結

原文鏈接:https://juejin.cn/post/7145856146740346916

欄目分類
最近更新