RecycleView报错Java.lang.IllegalArgumentException: Called attach on a child which is not detached

2019-01-27 21:24:03

最近项目的一个需求,一个银行卡列表页,银行卡初始状态是折叠,点击银行卡展开当前卡片,折叠上一张打开的银行卡,仅此而已,如下图效果: 



实现方式有很多种,我实现的一种思路就是:列表用Recyclerview,每个银行卡实体定义一个属性


boolean isopen = true/false //银行卡是否展开

1

根据这个打开或折叠银行卡,每次点击记录点击的position,并将它记录在Adapter的成员变量


int preIndex //上一个点击的item的索引

1

中,并设置另一个标记


boolean hasItemOpened //当前是否有卡片展开

1

每次点击时,首先根据当前item的isopen 属性来执行卡片的折叠或展开动作,然后再通过判断


if (preIndex != position && hasItemOpen)//说明当前点击的不是上一个卡片,并且上一张开片是展开状态

1

来处理上一张卡片的状态。


当然,这里重点不是如何来实现这个功能,而是在实现这个功能过程中发现的问题:

在处理上一张卡片的折叠动作时,我当时想,只需要改变数据源isopen 的值,再调用RecyclerView.Adapter的notifyItemChanged(int position)方法即可,但是程序出乎意料的挂了: 

错误如下: 

E/AndroidRuntime: FATAL EXCEPTION: mainProcess: com.paicaifu.riches, PID: 8502 

java.lang.IllegalArgumentException: Called attach on a child which is not detached: ViewHolder{adb21588 position=0 id=-1, oldPos=-1, pLpos:-1} 

at android.support.v7.widget.RecyclerView$5.attachViewToParent(RecyclerView.java:654) 

at android.support.v7.widget.ChildHelper.attachViewToParent(ChildHelper.java:239) 

at android.support.v7.widget.RecyclerView.addAnimatingView(RecyclerView.java:1107) 

at android.support.v7.widget.RecyclerView.animateChange(RecyclerView.java:3270) 

at android.support.v7.widget.RecyclerView.dispatchLayoutStep3(RecyclerView.java:3088) 

at android.support.v7.widget.RecyclerView.dispatchLayout(RecyclerView.java:2917) 

at android.support.v7.widget.RecyclerView.onLayout(RecyclerView.java:3283) 

… 

at android.view.Choreographer.doCallbacks(Choreographer.java:574) 

at android.view.Choreographer.doFrame(Ch



也没报哪一行,这个错误也没见过,不过这么多错误,这一行才是最主要的:Called attach on a child which is not detached: ViewHolder ,它的意思大概就是,操作的这个viewholder当前不是被绑定的(detached),这是为啥呢,经过查询得知,这才是RecyclerView缓存机制啊,未在屏幕上显示的item会被暂时收回,就是detached,所以当调用notifyItemChanged() 时,如果item不在屏幕中,就会导致这个错误,另外相关的还有一个知识点:

mLayoutManager.getChildCount(); 这个方法返回的并不是adapter所有item的数量,而是当前显示的item个数,详见源码:

     /**

         * Return the current number of child views attached to the parent RecyclerView.

         * This does not include child views that were temporarily detached and/or scrapped.

         *

         * @return Number of attached children

         */

        public int getChildCount() {

            return mChildHelper != null ? mChildHelper.getChildCount() : 0;

        }

1

2

3

4

5

6

7

8

9

好了,了解到这个问题以后,就开始解决吧,那么只需要判断当前position是否在屏幕中显示即可,哦了~不得不说Recyclerview是如此的强大,LayoutManager给我们提供了应有尽有的方法来判断position是否在屏幕中,通过LayoutManager我们可以获得当前屏幕显示的第一个item位置,以及最后一个item位置,这个位置是整个list中的位置 ,当满足position<=lastindex && position>=firstIndex 时,item即是现在在屏幕中,好了,这下可以安心的使用notifyItemChanged() 了,但是令人崩溃的是,程序又挂了,问题还是Called attach on a child which is not detached, 我的天,这次又是为何,经过debug发现,我的RecyclerView可以下啦刷新,position = 0的位置,是下拉刷新头,这个view肯定没有绑定ViewHolder啊,所以导致了以上的错误,好了,问题发现了,就解决吧,根据存储到成员变量preindex 中的值来计算要需要更新item的位置:int updatePos = preIndex - firstVisibleItemPosition+1, Ok,计算出了正确位置便可以通过以下方式更改上一个卡片的状态了:


int firstVisibleItemPosition = mLayoutManager.findFirstVisibleItemPosition();

            int lastVisibleItemPosition = mLayoutManager.findLastVisibleItemPosition();

            if (preIndex-firstVisibleItemPosition>=0&&preIndex<=lastVisibleItemPosition){

            //获取上一个卡片的view    

                View pre = mLayoutManager.getChildAt(preIndex - firstVisibleItemPosition+1);

                closeCard(pre);//折叠卡片

            }

1

2

3

4

5

6

7

OK,完成。虽然是很普通的一个功能,但是踩坑n多啊,总结一下,给自己也给各位android朋友一个提醒。如有错误之处,请不吝赐教,多多交流!


  • 2019-09-04 16:50:35

    CMake入门笔记

    Make是一个跨平台的安装(编译)工具,可以用简单的语句来描述所有平台的安装(编译过程)。他能够输出各种各样的makefile或者project文件,能测试编译器所支持的C++特性,类似UNIX下的automake。只是 CMake 的组态档取名为 CMakeLists.txt。Cmake 并不直接建构出最终的软件,而是产生标准的建构档(如 Unix 的 Makefile 或 Windows Visual C++ 的 projects/workspaces),然后再依一般的建构方式使用。这使得熟悉某个集成开发环境(IDE)的开发者可以用标准的方式建构他的软件,这种可以使用各平台的原生建构系统的能力是 CMake 和 SCons 等其他类似系统的区别之处。

  • 2019-09-05 20:51:15

    在Android上使用FFmpeg压缩视频

    libavcodec-提供了更加全面的编解码实现的合集 libavformat-提供了更加全面的音视频容器格式的封装和解析以及所支持的协议 libavutil-提供了一些公共函数 libavfilter-提供音视频的过滤器,如视频加水印、音频变声等 libavdevice-提供支持众多设备数据的输入与输出,如读取摄像头数据、屏幕录制 libswresample,libavresample-提供音频的重采样工具 libswscale-提供对视频图像进行色彩转换、缩放以及像素格式转换,如图像的YUV转换 libpostproc-多媒体后处理器

  • 2019-09-05 20:54:21

    在Android 中使用FFmpeg命令

    到这里就可以运行FFmpeg命令了。一直我也是这样使用,但是我在做这个项目Cut的时候发现连续调用多次FFmpeg命令会报错(在项目需要,先改变分镜头的速度,再合成视频)。 为什么会这样的呢?

  • 2019-09-06 10:30:20

    ffmpeg错误码

    AVERROR_BSF_NOT_FOUND = -1179861752 AVERROR_BUG = -558323010 AVERROR_DECODER_NOT_FOUND = -1128613112 AVERROR_DEMUXER_NOT_FOUND = -1296385272 AVERROR_ENCODER_NOT_FOUND = -1129203192 AVERROR_EOF = -541478725 AVERROR_EXIT = -1414092869 AVERROR_FILTER_NOT_FOUND = -1279870712 AVERROR_INVALIDDATA = -1094995529 AVERROR_MUXER_NOT_FOUND = -1481985528 AVERROR_OPTION_NOT_FOUND = -1414549496 AVERROR_PATCHWELCOME = -1163346256 AVERROR_PROTOCOL_NOT_FOUND = -1330794744 AVERROR_STREAM_NOT_FOUND = -1381258232 AVERROR_BUG2 = -541545794 AVERROR_UNKNOWN = -1313558101

  • 2019-09-08 09:05:54

    MyBatis Generator 插件的拓展插件包

    应该说使用Mybatis就一定离不开MyBatis Generator这款代码生成插件,而这款插件自身还提供了插件拓展功能用于强化插件本身,官方已经提供了一些拓展插件,本项目的目的也是通过该插件机制来强化Mybatis Generator本身,方便和减少我们平时的代码开发量。

  • 2019-09-08 09:09:48

    mybatis-generator自动生成代码插件使用详解

      mybatis-generator是一款在使用mybatis框架时,自动生成model,dao和mapper的工具,很大程度上减少了业务开发人员的手动编码时间,今天自己研究了一下,也分享一下使用心得供大家简单使用。