如果视图数量较少,RecyclerView无法回收视图。

9

我一直在处理Recycler View中的奇怪行为,如果适配器中的视图计数很小,例如我的情况是5个,则当将视图滚动回屏幕时,视图不会被回收利用,也不会调用onBindViewHolder。如果我将视图大小增加到10个,那么回收利用就开始工作了,并且每次视图进入屏幕时都会调用onBindViewHolder。

XML

<android.support.v4.widget.SwipeRefreshLayout
    android:id="@+id/swipeRefresh"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content">

    <!-- RECYCLER VIEW -->
    <android.support.v7.widget.RecyclerView
        android:id="@+id/eventsList"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</android.support.v4.widget.SwipeRefreshLayout>

Recycler View 初始化

    LinearLayoutManager llm = new LinearLayoutManager(getActivity());
    llm.setOrientation(LinearLayoutManager.VERTICAL);

    eventsList.setLayoutManager(llm);

    //Sets endless listener
    scrollListener = new EndlessRecyclerOnScrollListener(llm,3,mLastDownloadedPage) {
        @Override
        public void onLoadMore(int current_page) {
            Timber.d("Loading date for page: " + current_page);

            mLastDownloadedPage = current_page;

            //If it's showing cached don't download
            if(showingCached)
                return;

            //Shows snackbar loading view
            showSnackbarLoadingView();

            //Gets events
            getEvents(false);

        }
    };

    //eventsList.addOnScrollListener(scrollListener);

    //Set the adapter
    mAdapter = new EventListAdapter(this,getActivity().getApplicationContext());

    eventsList.setAdapter(mAdapter);

适配器
@Override
public EventHolder onCreateViewHolder(ViewGroup viewGroup, int i) {

    Timber.d("Create view holder pos: "+i);

    View v = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.list_events_row, viewGroup, false);
    return new EventHolder(v,mListener);

}

@Override
public void onBindViewHolder(EventHolder eventHolder, int i) {

    if(mBindListener != null)
        mBindListener.onBind(eventHolder,i);

    Timber.d("Bind view holder pos "+i);

    Event event = eventList.get(i);
    eventHolder.bindEvent(event);

    //Sets the date
    setDate(eventHolder.date, event);

    //Sets the location
    setLocation(eventHolder.distance,event);


}

屏幕上只有3个可见视图,大小为5,当我向下滚动时,我可以看到createViewHolderbindViewHolder被调用的位置是4和5,但当我向上滚动时,它们在退出屏幕时不会被回收。我可以确认这一点,因为通过覆盖

@Override
public void onViewRecycled(EventHolder holder) {
    super.onViewRecycled(holder);
    Timber.d("onViewRecycled: "+holder.name);
}

如果视图计数增加,例如大小为10,则不记录任何消息。但是,一切正常,视图被回收,并且对于每个位置都会调用onBindViewHolder

这是正常行为吗?如果是,我如何保存项目视图的状态,例如激活状态?我曾经将活动位置保存在变量中,然后使用此行在进入屏幕时激活视图,使用绑定方法。

        holder.itemView.setActivated(pos == lastSelectedPos);
1个回答

22

这是正常行为,但你可以通过以下方式进行微调:

recyclerView.setItemViewCacheSize(int);

关于问题的第二部分 - 你是对的!将“已激活”位置存储为适配器中的变量。在onBindViewHolder()方法中执行以下操作:

holder.itemView.setActivated(holder.getAdapterPosition() == lastSelectedPos);

23
提一下,当我深入了解 recyclerView.setItemViewCacheSize() 方法时,缓存的默认值为 2 - kirtan403
谢谢kirtan403,我已经到处找过了。 - esfox
3
救星!只需要设置recyclerView.setItemViewCacheSize(-1);,它就像魔法一样正常工作了!非常感谢。 注意:当设置recyclerView.setItemViewCacheSize(0);时,第一个项目会出现问题,所以将其更改为-1并解决了问题。 - Hanoch Moreno

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接