RecyclerView踩坑笔记

问题一.RecyclerView中嵌套另外一个RecyclerView时,内部的RecyclerView滑动时状态是不会变化的。

原因:

解决方案:获取父级的RecyclerView进行判断。

 

问题二:发现onCreateViewHolder频繁被调用,从而导致卡顿。

原因:页面刷新较为频繁。 所以每次notifyDataSetChanged提示刷新的时候,会回收掉所有已经展示的ViewHolder,然后重新渲染,RecyclerView默认的缓存数量是5个。这时候,如果当前展示的数量超过5个,比如8个。那么就会有3个ViewHolder找到可复用的对象,从而走onCreateViewHolder的流程,初始化View的操作是十分耗时的,而且是主线程,从而导致卡顿。

解决方案:如果刷新确实比较频繁的话,则按照当前页面估算能展示多少个ViewHolder,扩大复用的数值,比如我就把ViewHolder的复用数量改为了10。

itemView.rv_list.recycledViewPool.setMaxRecycledViews(0, 10)

 

问题三:java.lang.IndexOutOfBoundsException: Inconsistency detected. Invalid view holder adapter positionViewHolder

原因:滑动的过程中,如果依赖的list的数量发生变化,则就会抛出这样的错误。

解决方案:

1.确保adapter依赖独立的数据源。这个意思就是说adapter独占一个recycleList,而进行操作的是另外一个dataList。当需要刷新时,在主线程中,清空recycleList,把数据源dataList加到recycleList中。并且立即调用notifyDataSetChanged进行刷新。

示例如下:

functionItemAdapter?.recyclerList?.clear()
functionItemAdapter?.recyclerList?.addAll(mDataList)
functionItemAdapter?.notifyDataSetChanged()

 

问题四:滑动过程中卡顿

原因:调研发现,我们的需求模型下,数据源变动十分的频繁,1秒钟变动两三次,所以就造成RecyclerView频繁处于绘制状态。

绘制是占用主线程的,所以这时候滑动的话,滑动触发的绘制会进行排队,从而导致给人感觉是滑动卡顿。

解决方案:

1.考虑到用户的实际体验,滑动过程中用户是不会care数据的实时性的。所以我们在滑动的时候,可以先把传入的变更数据缓存起来,此时只是缓存数据,并不刷新。等到监控到滑动结束的时候,使用缓存数据和新传入的数据拼装后一起通知刷新。

 

问题五:RecyclerView下滑时崩溃

原因:我使用的是XRecyclerView框架,拖动往下滑动的时候,就会发生崩溃。

onRefresh的方法里,清空了adapter中的数据,这时候,并没有调用notifyDataSetChanged进行刷新。往下拖动其实类似于一个动画,也是需要数据进行刷新的。这时候数据被清空了,自然就没有数据进行刷新,则抛出异常。

这篇文章有比较详细的解释:https://www.jianshu.com/p/4fb94744f61b

崩溃日志:

2020-07-24 14:54:36.860 27847-27847/com.commonrail.mft.decodertest E/Cockroach: java.lang.IllegalArgumentException: Scrapped or attached views may not be recycled. isScrap:false isAttached:true com.jcodecraeer.xrecyclerview.XRecyclerView{169d628 VFED..... ......ID 0,0-1875,405 #7f0802de app:id/rv_brush_list}, adapter:com.jcodecraeer.xrecyclerview.XRecyclerView$WrapAdapter@4408f41, layout:androidx.recyclerview.widget.LinearLayoutManager@868fce6, context:com.commonrail.mft.decoder.ui.brushData.BrushDataActivity@494f176
        at androidx.recyclerview.widget.RecyclerView$Recycler.recycleViewHolderInternal(RecyclerView.java:6433)
        at androidx.recyclerview.widget.RecyclerView$Recycler.quickRecycleScrapView(RecyclerView.java:6554)
        at androidx.recyclerview.widget.RecyclerView$LayoutManager.removeAndRecycleScrapInt(RecyclerView.java:9249)
        at androidx.recyclerview.widget.RecyclerView.dispatchLayoutStep3(RecyclerView.java:4207)
        at androidx.recyclerview.widget.RecyclerView.dispatchLayout(RecyclerView.java:3862)
        at androidx.recyclerview.widget.RecyclerView.onLayout(RecyclerView.java:4404)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1816)
        at android.widget.LinearLayout.layoutHorizontal(LinearLayout.java:1805)
        at android.widget.LinearLayout.onLayout(LinearLayout.java:1571)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1816)
        at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1660)
        at android.widget.LinearLayout.onLayout(LinearLayout.java:1569)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.FrameLayout.layoutChildren(FrameLayout.java:325)
        at android.widget.FrameLayout.onLayout(FrameLayout.java:261)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1816)
        at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1660)
        at android.widget.LinearLayout.onLayout(LinearLayout.java:1569)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1816)
        at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1660)
        at android.widget.LinearLayout.onLayout(LinearLayout.java:1569)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1816)
        at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1660)
        at android.widget.LinearLayout.onLayout(LinearLayout.java:1569)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.FrameLayout.layoutChildren(FrameLayout.java:325)
        at android.widget.FrameLayout.onLayout(FrameLayout.java:261)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1816)
        at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1660)
        at android.widget.LinearLayout.onLayout(LinearLayout.java:1569)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.FrameLayout.layoutChildren(FrameLayout.java:325)
        at android.widget.FrameLayout.onLayout(FrameLayout.java:261)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1816)
        at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1660)
        at android.widget.LinearLayout.onLayout(LinearLayout.java:1569)
        at android.view.View.layout(View.java:19781)
        at android.view.ViewGroup.layout(ViewGroup.java:6144)
        at android.widget.FrameLayout.layoutChildren(FrameLayout.java:325)
    	at android.widget.FrameLayout.

解决方案:

等数据回来后直接进行数据源的替换,然后通知刷新。而不提前进行数据源清空操作。

你可能感兴趣的:(安卓)