前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >当你刷新RecyclerView程序崩掉的时候

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

作者头像
代码咖啡
发布2018-08-28 10:04:04
2.2K0
发布2018-08-28 10:04:04
举报
文章被收录于专栏:程序员叨叨叨程序员叨叨叨

崩溃再现

今天测试 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#

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2016.12.30 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 崩溃再现
  • 报错信息
  • 寻找解决方法
  • 找到真正原因
  • 参考:
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档