美文网首页android技术专栏Android知识Android开发
当你刷新RecyclerView程序崩掉的时候

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

作者: 代码咖啡 | 来源:发表于2016-12-30 17:31 被阅读10816次

    崩溃再现

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

    相关文章

      网友评论

        本文标题:当你刷新RecyclerView程序崩掉的时候

        本文链接:https://www.haomeiwen.com/subject/wspmvttx.html