版权声明:本文为博主原创文章,未经博主允许不得转载。
记录下碰到的问题,在做仿饿了么商店效果的时候,碰到了个比较扎心的bug,当商品列表的recycleview已经加载完了,手动滑动rv列表,这个时候去点击商品类型,导致了程序崩溃.日志如下:
第一步:定位问题
对着log看了一眼,发现了IndexOutOfBoundsException这个异常,联想到崩溃产生时的场景,不难想到滑动的时候又请求数据才导致的,但是问题又来了,接着往下看log,并没有项目的报错位置信息,觉得应该是recycleview原生的问题.
第二步:解决问题
在网上查了下相关资料,总结下几种解决方式:
1:自定义LinearLayoutManager,复写supportsPredictiveItemAnimations,返回false,是让recycleview不执行相关部分动画
2:在获取到新数据后再清空数据(请求到新数据后再清空旧数据)
3:在执行notifyDataSetChanged之前,现判断当前时候还在滑动,如果没有滑动,执行notifyDataSetChanged方法。
if (mRecyclerView.getScrollState() == RecyclerView.SCROLL_STATE_IDLE ||
!mRecyclerView.isComputingLayout()) {
mAdapter.notifyDataSetChanged();
}
4:重写RecyclerView的OnTouchListener,在刷新时用一个布尔值去处理列表是否可以滑动.
这几个方法对于某些场景下应该还是有效果的,但是对于我来说,没有解决根本问题.在滑动的时候,刷新数据,导致数据的list的size为0,当此时滑动就会导致IndexOutOfBoundsException,同理,在rv滑动的时候去请求数据,也会导致这个问题,原理都是越界了.
针对标题这个问题,其实只要clear旧数据之后刷新rv,再将新的数据显示在rv上,最后再刷新rv就可以了.这是让rv在list的size为0的时候知道了这个变化,重置了list的大小,等新数据重新获取后再刷新,便能得到最新的数据了,崩溃的问题也不会出现.
解决办法如下(以上4中方法不能解决问题可以试试这个):
clear()//清空旧list
notifyDataSetChanged()//刷新数据
add()//增加新数据
notifyDataSetChanged()//刷新数据