RecyclerView 解析

在项目中使用RecyclerView已经有很长一段时间了,现在结合RecyclerView源码分析总结一下。
RecyclerView:是support-v7包中的新组件,用于在有限的窗口显示大量的数据,项目中之前使用到的ListView、GridView等可以轻易的替换为RecyclerView,这里强力推荐大家使用RecyclerView。

RecyclerView直接继承于ViewGroup,提供很多方便实用的public方法,提供有很多内部类,基本代码结构如下:

public class RecyclerView extends ViewGroup implements ScrollingView, NestedScrollingChild {
    ...
    final Recycler mRecycler = new Recycler();

    private Adapter mAdapter;
    private LayoutManager mLayout;
    ...
    public RecyclerView(Context context) {
        this(context, null);
    }

    public RecyclerView(Context context, @Nullable AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public RecyclerView(Context context, @Nullable AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        ...
    }

    public void setLayoutManager(LayoutManager layout) {
    ...
    }

    ...
    public void setItemViewCacheSize(int size) {
        mRecycler.setViewCacheSize(size);
    }

    public void addItemDecoration(ItemDecoration decor, int index) {
    ...
    }

    ...
    public static class RecycledViewPool {
    ...
    }

    public final class Recycler {
    ...
    }

    public abstract static class ViewCacheExtension {
    ...
    }

    public static abstract class Adapter<VH extends ViewHolder> {
    ...
    }

    public static abstract class LayoutManager {
    ...
    }

    public static abstract class ItemDecoration {
    ...
    }

    public static abstract class ViewHolder {
    ...
    }

    public static abstract class ItemAnimator {
    ...
    }
    ...
}

下面挑选一些重要的内部类简单介绍一下:

RecyclerView.Adapter: 提供具体应用数据和RecyclerView展示的条目View的绑定。
这里的Adapter和以往ListView、GridView等使用到的Adapter完全不同。ListView、GridView等View的Adapter这里不做详细介绍。

RecyclerView.Adapter的代码结构如下:

public static abstract class Adapter<VH extends ViewHolder> {
        ...

        public abstract VH onCreateViewHolder(ViewGroup parent, int viewType);

        public abstract void onBindViewHolder(VH holder, int position);

        public final VH createViewHolder(ViewGroup parent, int viewType) {
            TraceCompat.beginSection(TRACE_CREATE_VIEW_TAG);
            final VH holder = onCreateViewHolder(parent, viewType);
            holder.mItemViewType = viewType;
            TraceCompat.endSection();
            return holder;
        }

        public final void bindViewHolder(VH holder, int position) {
            holder.mPosition = position;
            if (hasStableIds()) {
                holder.mItemId = getItemId(position);
            }
            holder.setFlags(ViewHolder.FLAG_BOUND,
                    ViewHolder.FLAG_BOUND | ViewHolder.FLAG_UPDATE | ViewHolder.FLAG_INVALID
                            | ViewHolder.FLAG_ADAPTER_POSITION_UNKNOWN);
            TraceCompat.beginSection(TRACE_BIND_VIEW_TAG);
            onBindViewHolder(holder, position);
            TraceCompat.endSection();
        }        

        public int getItemViewType(int position) {
            return 0;
        }

        public abstract int getItemCount();

        ...
}

阅读RecyclerView的源码,发现其Adapter没有任何默认实现,在使用的时候需要自己实现Adapter,三个抽象函数是必须要实现的,getItemViewType函数可根据实际需求进行重载,如果RecyclerView的Item有多种类型,则需要重载,否则默认实现方式就可以了。

RecyclerView.ViewHolder
代表RecyclerView的每一个ItemView,通过它保存Item中使用到的控件的引用来减少findViewById的调用,RecyclerView管理ItemView的重用回收等也是通过存贮iewHolder来实现。

    public static abstract class ViewHolder {
        public final View itemView;
        ...
        public ViewHolder(View itemView) {
            if (itemView == null) {
                throw new IllegalArgumentException("itemView may not be null");
            }
            this.itemView = itemView;
        }   
        ...     

正如源码所示,itemView 已经以public 方式存放了,自定义ViewHolder 时,不再需要在重复保存itemView ,项目中需要对itemView 操作(如设置背景、添加事件等),直接使用就可以了。
实际使用的时候,使用自定义的ViewHolder 来替换RecyclerView.Adapter里面的泛型VH。

RecyclerView.LayoutManager
LayoutManager 负责测量和放置itemView 在RecyclerView合适的位置,决定不可见的itemView 的回收方式。通过改变LayoutManager ,RecyclerView可以实现垂直滚动列表,网格,瀑布流,水平滚动列表等等,使用方式非常灵活,高效。android.support.v7库已经默认实现了几种LayoutManager ,项目使用中有特殊需求也可以自定义LayoutManager 。
自定义LayoutManager 需要分别重载和实现如下两个函数:

        public void onLayoutChildren(Recycler recycler, State state) {
            Log.e(TAG, "You must override onLayoutChildren(Recycler recycler, State state) ");
        }

        public abstract LayoutParams generateDefaultLayoutParams();

android.support.v7库提供的LayoutManager 默认实现有如下几种:

import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.StaggeredGridLayoutManager;

其中,LinearLayoutManager提供类似ListView的功能,可以实现水平或垂直列表,默认为垂直列表。

public class LinearLayoutManager extends RecyclerView.LayoutManager implements
        ItemTouchHelper.ViewDropHandler {
    public LinearLayoutManager(Context context) {
        this(context, VERTICAL, false);
    }  

    public LinearLayoutManager(Context context, int orientation, boolean reverseLayout) {
        setOrientation(orientation);
        setReverseLayout(reverseLayout);
    }

    public LinearLayoutManager(Context context, AttributeSet attrs, int defStyleAttr,
                               int defStyleRes) {
        Properties properties = getProperties(context, attrs, defStyleAttr, defStyleRes);
        setOrientation(properties.orientation);
        setReverseLayout(properties.reverseLayout);
        setStackFromEnd(properties.stackFromEnd);
    }          
}

GridLayoutManager 提供类似GradView的功能,实现表格布局。

StaggeredGridLayoutManager 实现瀑布流。

RecyclerView.ItemDecoration
ItemDecoration可以用来添加items分割线,让item高亮,给组添加边界等等。

RecyclerView.ItemAnimator
ItemAnimator 提供Item添加、删除时的动画效果。
android.support.v7 提供了一个默认实现DefaultItemAnimator,如果没有设置动画效果,RecyclerView使用默认的DefaultItemAnimator。实际使用的时候可以扩展ItemAnimator,实现需要的,符合要求的,非常炫的动画效果。

RecyclerView.RecycledViewPool
RecycledViewPool 可以实现多个RecyclerViews共享Views,有效减少创建ViewHolder的开销,避免垃圾回收。实际项目使用中,如果你没有提供RecycledViewPool,RecyclerViews也会自己默认自动创建一个。
查看RecyclerView的源码

        ...
        void setRecycledViewPool(RecycledViewPool pool) {
            if (mRecyclerPool != null) {
                mRecyclerPool.detach();
            }
            mRecyclerPool = pool;
            if (pool != null) {
                mRecyclerPool.attach(getAdapter());
            }
        }

        RecycledViewPool getRecycledViewPool() {
            if (mRecyclerPool == null) {
                mRecyclerPool = new RecycledViewPool();
            }
            return mRecyclerPool;
        }
        ...

通过上述源码,发现RecycledViewPool使用也非常简单,只需获取任意一个RecyclerView创建的RecycledViewPool,然后分别设置到其他RecyclerView,就可实现共享。

        RecyclerView view1 = new RecyclerView(mContext);
        LinearLayoutManager layout = new LinearLayoutManager(mContext);
        layout.setRecycleChildrenOnDetach(true);
        view1.setLayoutManager(layout);
        RecyclerView.RecycledViewPool recycledViewPool = view1.getRecycledViewPool();

        RecyclerView view2 = new RecyclerView(mContext);
        view2.setRecycledViewPool(recycledViewPool);
        ...

        RecyclerView view3 = new RecyclerView(mContext);
        view3.setRecycledViewPool(recycledViewPool);
        ...

RecyclerView 简单使用如下:

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.my_recyclerview_test);
        ...
        RecyclerView recyclerView = (RecyclerView) findViewById(R.id.recycler_view);
        RecyclerView.LayoutManager layoutManager = new LinearLayoutManager(mContext);
        recyclerView.setLayoutManager(layoutManager);
        MyRecyclerViewAdapter adapter = new MyRecyclerViewAdapter();
        recyclerView.setAdapter(adapter);
        ...
    }

    ...
    public class MyRecyclerViewHolder extends RecyclerView.ViewHolder {
        public TextView textView;
        public MyRecyclerViewHolder(View itemView) {
            super(itemView);
            textView = (TextView) itemView.findViewById(R.id.name);
        }
    }
    public class MyRecyclerViewAdapter extends RecyclerView.Adapter<MyRecyclerViewHolder> {

        @Override
        public MyRecyclerViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
            View view = LayoutInflater.from(mContext).inflate(R.layout.item_recyclerview_test_layout, parent, false);
            MyRecyclerViewHolder viewHolder = new MyRecyclerViewHolder(view);
            return viewHolder;
        }

        @Override
        public void onBindViewHolder(MyRecyclerViewHolder holder, int position) {
            holder.textView.setText("hello "+position);
        }

        @Override
        public int getItemCount() {
            return 3;
        }
    }

总结,RecyclerView代码耦合非常低,具有如下优点:
1、RecyclerView里View的布局完全由LayoutManager管理,布局灵活高效,本身提供了常用布局,也可以简单的通过扩展实现想要的布局方式;
2、RecyclerView对View的回收重用管理比传统ListView、GradeView等更加高效、灵活;
3、对ItemView的添加装饰如设置分割线、背景、高亮等非常灵活、简单;
4、RecyclerView添加ItemView变化(数据增加、删除等)时的动画非常灵活,而且本身默认就提供了动画,也可以扩展实现非常炫的动画效果。
。。。

当然,RecyclerView源码里还有很多细节,实际使用的时候,结合源码,通过扩展RecyclerView,可以实现非常高效、炫丽的List效果。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值