今天遇到的一个Bug,这里纪录一下,如果下次再遇到,可以直接拿来用。这个算是Android官方的一个Bug,迟迟没有解决,网上可以看到一大群人遇到该问题,说白了就是还没有绘制完不能滑动限制住了就可以了。
报错信息:
java.lang.IndexOutOfBoundsException: Inconsistency detected. Invalid item position 21(offset:21.state:20
at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:3300)
at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:3258)
at android.support.v7.widget.LinearLayoutManager$LayoutState.next(LinearLayoutManager.java:1803)
at android.support.v7.widget.LinearLayoutManager.layoutChunk(LinearLayoutManager.java:1302)
at android.support.v7.widget.LinearLayoutManager.fill(LinearLayoutManager.java:1265)
at android.support.v7.widget.LinearLayoutManager.scrollBy(LinearLayoutManager.java:1093)
at android.support.v7.widget.LinearLayoutManager.scrollVerticallyBy(LinearLayoutManager.java:956)
at android.support.v7.widget.RecyclerView$ViewFlinger.run(RecyclerView.java:2715)
at android.view.Choreographer$CallbackRecord.run(Choreographer.java:725)
at android.view.Choreographer.doCallbacks(Choreographer.java:555)
at android.view.Choreographer.doFrame(Choreographer.java:524)
at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:711)
at android.os.Handler.handleCallback(Handler.java:615)
at android.os.Handler.dispatchMessage(Handler.java:92)
at android.os.Looper.loop(Looper.java:137)
at android.app.ActivityThread.main(ActivityThread.java:4921)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:511)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1027)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:794)
at dalvik.system.NativeStart.main(Native Method)
在Recyclerview滑动的时候,更改了Adapter的Data,执行notifyDataSetChanged()导致的。
思路一:自己继承LinearLayoutManager复写下面的方法,返回false,思路是不执行部分动画,以及对super.onLayoutChildren(recycler, state);捕获异常
@Overridepublic boolean supportsPredictiveItemAnimations() {
return false;
}
@Override
public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {
try {
super.onLayoutChildren(recycler, state);
} catch (IndexOutOfBoundsException e) {
Log.e("bug","crash in RecyclerView");
}
}
然而然并卵,依然会crash。
思路二:数据先clear然后执行otifyDataSetChanged,然后add后执行notifyDataSetChanged,然并卵,不管你什么时候执行,都会crash
思路三:刷新的时候不让用户滑动。这种方式肯定是可以的,屏蔽掉了滑动事件,肯定不会有这个冲突。
mRecyclerView.setOnTouchListener(
new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
if (mIsRefreshing) {
return true;
} else {
return false;
}
}
}
);
//当刷新时设置
//mIsRefreshing=true;
//刷新完毕后还原为false
//mIsRefreshing=false;
但是代价太大,太影响用户体验。产品肯定不会同意,Pass。
解决方案一:在执行notifyDataSetChanged之前,现判断当前是否还在滑动,如果没有滑动,执行notifyDataSetChanged方法。
if (mRecyclerView.getScrollState() == RecyclerView.SCROLL_STATE_IDLE ||
!mRecyclerView.isComputingLayout()) {
mAdapter.notifyDataSetChanged();
}
这种方式在用户没有滑动操作的时候刷新,有滑动,也不用怕,因为数据已经替换为线上返回的数据,在用户滑动的时候依然会刷新。
解决方案二:思路和方案一差不多,但是用递归的方式保证更新代码肯定会执行不会导致有时候数据没有刷新的情况
/**
* 递归等待滑动停止后再删除数据
*
* @param deleteData 不建议使用position,因为如果出现多个等待position将会出错
*/
public void waitRefreshData(final XXData data) {
if (mRecyclerView.isComputingLayout()) {
mRecyclerView.postDelayed(new Runnable() {
@Override
public void run() {
waitRefreshData(data);
}
}, 100);
} else {
//必须先clear()之后刷新一下,然后再填充数据刷新,不然会有问题
data.clear();
mAdapter.notifyDataSetChanged();
mAdapter.setData(data);
mAdapter.notifyDataSetChanged();
}
}