当你刷新RecyclerView程序崩掉的时候

崩溃再现

今天测试 APP的时候发现一个有关RecyclerView的BUG,我们先上图来看看崩溃情况:

崩溃再现

报错信息

摸着后脑勺,一脸蒙圈地看着Android Studio打印着下面这段日志:

java.lang.IllegalArgumentException: Scrapped or attached views may not be recycled. isScrap:false isAttached:true
at android.support.v7.widget.RecyclerView$Recycler.recycleViewHolderInternal(RecyclerView.java:5659)
    at android.support.v7.widget.RecyclerView$Recycler.recycleView(RecyclerView.java:5603)
    at android.support.v7.widget.GapWorker.prefetchPositionWithDeadline(GapWorker.java:277)
    at android.support.v7.widget.GapWorker.flushTaskWithDeadline(GapWorker.java:324)
    at android.support.v7.widget.GapWorker.flushTasksWithDeadline(GapWorker.java:337)
    at android.support.v7.widget.GapWorker.prefetch(GapWorker.java:344)
    at android.support.v7.widget.GapWorker.run(GapWorker.java:370)
    at android.os.Handler.handleCallback(Handler.java:743)
    at android.os.Handler.dispatchMessage(Handler.java:95)
    at android.os.Looper.loop(Looper.java:150)
    at android.app.ActivityThread.main(ActivityThread.java:5665)
    at java.lang.reflect.Method.invoke(Native Method)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:822)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:712)

寻找解决方法

定眼一看,这个信息并不是传统型经典款的异常,不像空指针异常,找不到出错的地方。无奈之下,寻求万能的谷歌。通过查询,得到如下两种原因及解决方案:

  • android:animateLayoutChanges与RecyclerView刷新共用

图片截自StackOverFlow 通过蹩脚的翻译,得到如下信息:

这个错误的原因是xml布局文件中将android:animateLayoutChanges 设置为true 并且java 代码里对RecyclerViewadapter 调用了notifyDataSetChanged() 方法。

由此可知晓android:animateLayoutChanges="true"与RecyclerView的刷新不可同时存在,那么为什么同时存在会报错呢? 我们知道android:animateLayoutChanges="true"是在列表增删Item的时候调用系统自带的动画效果,而RecyclerView的机制是对子视图的复用,不会在真正意义上对Item的控件进行增加、删除,与android:animateLayoutChanges原理相悖,因此会报错。

  • item中获取焦点的控件使之无法销毁

图片截自StackOverFlow 从回答中,我们可以看到,在多个Item包含EditText的时候,因为其中一个EditText抢占了RecyclerView的焦点,导致无法被回收,所以报错。其解决方法为给RecyclerView添加focusableInTouchMode属性,使之在触摸模式下(手指接触屏幕)依然可以获取到焦点,从而对Item进行回收刷新。

找到真正原因

尝试了上述两种方法,发现问题依然没有解决,因问题出现在刷新的地方,而我使用的是XRecyclerView框架,所以到onRefresh方法里去找原因,发现了问题的真正原因:

onRefresh操作代码

从上述代码中可以看到,我先清空了已有的数据,然后再获取网络接口的数据,而从清空数据到获取到数据的这段时间里,List中的数据是不存在的,所以给了RecyclerView要回收Item,但是View没有被回收的假象(此时并没有执行notifyDataSetChanged()方法),因此程序报错。

其解决方法是:将列表清空的方法放到获取到接口数据以后执行。

以上就是本文的所有内容,对于上述解决方法的理解存在“主观臆断”的倾向,因此本文仅供参考,若有错误的地方,欢迎大家在文章下方评论指正!

参考:

http://blog.csdn.net/io_field/article/details/53083586 http://stackoverflow.com/questions/26477660/recyclerview-crashes-when-scrapped-or-attached-views-may-not-be-recycled#

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏GopherCoder

『No19: Gorm 上手指南』

如果你是做后端开发的,日常工作中,除了熟悉编程语言之外,数据库怕是最常用的技术了吧。

6711
来自专栏点滴积累

Cesium中Clock控件及时间序列瓦片动态加载

前言 前面已经写了两篇博客介绍Cesium,一篇整体上简单介绍了Cesium如何上手,还有一篇介绍了如何将Cesium与分布式地理信息处理框架Geotrelli...

4504
来自专栏以南小隐-数通那些事儿

锐捷RSOS_10.4之后版本Ctrl层升级系统版本

1804
来自专栏高性能服务器开发

从零学习开源项目系列(四)LogServer源码探究

这是从零学习开源项目的第四篇,上一篇是《从零学习开源项目系列(三) CSBattleMgr服务源码研究》,这篇文章我们一起来学习LogServer,中文意思可能...

2562
来自专栏Core Net

懒人小技巧, Toad 常用偷懒方法

3359
来自专栏小灰灰

Java & PhantomJs 实现html输出图片

Java & PhantomJs 实现html输出图片 借助phantomJs来实现将html网页输出为图片 I. 背景 如何在小程序里面生成一张图,分享到朋...

7148
来自专栏杨建荣的学习笔记

关于ORA-01779问题的分析和解决 (r4笔记第22天)

最近同事问我一个问题,是关于一个update语句的问题,需求有点特别,结果在使用update语句尝试了各种方法后,仍然是不依不饶的报出ORA-01779的错误。...

3507
来自专栏你不就像风一样

网络爬虫之Url含有中文如何转码

1112
来自专栏武军超python专栏

2018年8月18日初识tkinter

把C盘里面的东西移动到其他盘对文件有影响吗?普通文件如音频视频没有影响,但是如果是软件的话 下载的时候会在注册表中记录打开文件的路径,如果移动到其他盘的话注册...

1102
来自专栏腾讯Bugly的专栏

不给“爸爸”添麻烦 - iTOP iOS 动态库改造

苹果官方文档 对提交商店 APP 的二进制文件中__TEXT段大小有限制,超过大小限制的应用在提交评审的时候会被拒绝...

5969

扫码关注云+社区

领取腾讯云代金券