在使用RecylerView 的notifyItemChanged(int itemPosition)方法时,由于缓存引起的异常

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

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

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

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

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

      Called attach on a child which is not detached意思大概是,当前viewholder不是被绑定的(detached)。

由于RecyclerView缓存机制,未在屏幕上显示的item会被暂时收回,即detached,所以当调用notifyItemChanged() 时,如果item不在屏幕中,就会导致Called attach on a child which is not detached错误,还有另外一个相关的知识点:mLayoutManager.getChildCount(); 方法返回的是当前显示的item个数,并不是adapter所有item的数量。

      既然明白问题的出处,那需要判断当前position是否在屏幕中显示即可,LayoutManager给我们提供了相应的方法来判断position是否在屏幕中,我们可以通过LayoutManager获得当前屏幕显示的第一个item位置,以及最后一个item位置,注意该位置是整个list中的位置 ,当满足position<=lastindex && position>=firstIndex 时,说明item在屏幕中,即可安心的使用notifyItemChanged() 。如果RecyclerView可以下拉刷新或添加头部视图,position = 0的位置可能是下拉刷新视图或头部视图,那么需要根据存储到成员变量preindex 中的值来计算要需要更新的item位置:int updatePos = preIndex - firstVisibleItemPosition+1,即正确位置。

示例:

//第一个可见的位置

  int firstVisiblePosition = ((LinearLayoutManager) mRecyclerViewOrder.getLayoutManager()).findFirstVisibleItemPosition();

//头部不跟新

  firstVisiblePosition = firstVisiblePosition == 0 ? (firstVisiblePosition + 1) : firstVisiblePosition;

//最后一个可见的位置

    int lastVisiblePosition = ((LinearLayoutManager) mRecyclerViewOrder.getLayoutManager()).findLastVisibleItemPosition();

    if (mItemPosition - firstVisiblePosition >= 0 && mItemPosition <= lastVisiblePosition)

                myOrderTradeAdapter.notifyItemChanged(mItemPosition);//更新数据

你可能感兴趣的:(在使用RecylerView 的notifyItemChanged(int itemPosition)方法时,由于缓存引起的异常)