DiffResult调度导致检测到不一致。无效的视图持有者适配器positionViewHolder'有时会出错

时间:2017-10-04 11:17:08

标签: android android-recyclerview adapter android-viewholder

我有一个RxJava2 Observable,它接受两个列表,为它们计算diff结果并将此数据发送到适配器。主线程上的适配器调度更新。

适配器中的调度代码:

 public void dispatchStreams(List<StreamV3> streams, @Nullable DiffUtil.DiffResult diffResult) {

    if (streams == null) return;

    streamsList.clear();
    streamsList.addAll(streams);

    if (diffResult != null) {
        diffResult.dispatchUpdatesTo(this);
    }
}

我发现了“不一致”。无效的视图持有者适配器positionViewHolder&#39;某些设备上的错误。我无法弄清楚我的代码有什么问题。 Min SDK 21,Target SDK 26,RecyclerView版本为26.0.0。我知道扩展LinearLayoutManager并默默地捕获此错误的解决方法,但这是一个糟糕的解决方案,我相信这里应该更好。

有人可以提供任何帮助吗?

2 个答案:

答案 0 :(得分:9)

我在此answer

中找到了此问题的解决方案

似乎问题是由布局管理器上的supportsPredictiveItemAnimations属性引起的。当我将其设置为false时,不再发生崩溃。

public class LinearLayoutManagerWrapper extends LinearLayoutManager {

 public LinearLayoutManagerWrapper(Context context) {
   super(context);
 } 

 public LinearLayoutManagerWrapper(Context context, int orientation, boolean reverseLayout) {
   super(context, orientation, reverseLayout);
 }

 public LinearLayoutManagerWrapper(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
   super(context, attrs, defStyleAttr, defStyleRes);
 }

 @Override
 public boolean supportsPredictiveItemAnimations() {
   return false;
 }
}

答案 1 :(得分:0)

在编写要在后台线程中运行的Rx RV Diff时要注意此问题。将supportsPredictiveItemAnimations设置为false是一种防止崩溃的解决方法,但并不能真正解决问题。

在我的案例中导致此异常的原因是后台线程中数据集的突变。

// Diff and update the RV asynchronously
fun update(list: List<D>) {
    Observable
        .create<Pair<List<D>, DiffUtil.DiffResult>> {
            // runs it asynchronous
            val result = DiffUtil.calculateDiff(
                diffCallback.apply {
                    newList = list
                }
            )

            it.onNext(Pair(diffCallback.newList, result))
            it.onComplete()
        }
        .takeUntil(destroyObservable) // observe the Lifecycle of the Frag
        .subscribeOn(Schedulers.computation()) // run it async
        .observeOn(AndroidSchedulers.mainThread()) // jump to the main thread
        .subscribe {
            // Set the new list
            dataSet = it.first.map { it }
            it.second.dispatchUpdatesTo(this@ListComponentAdapter)
        }
}