美文网首页
Android控件RecycleView的基本用法

Android控件RecycleView的基本用法

作者: 卖火柴的笨小孩 | 来源:发表于2018-12-04 22:34 被阅读34次

    前言:虽然在日常开发中已经多次接触过RecycleView,但也只是用到其最基本的功能,并没有深入研究其他内容。接下来将抽出时间去了解RecycleView的相关内容,同时在博客中进行记录,以此加深印象。这篇文章主要是介绍RecycleView的使用方法。

    一、RecycleView是什么

           RecycleView是Android5.0后谷歌推出的一个用于在有限的窗口中展示大量数据集的控件,位于support-v7包中。它可以实现与ListView和GridView一样的效果,提供了一种插拔式的体验,高度的解耦,异常的灵活,只需设置其提供的不同的LayoutManager,ItemAnimator和ItemDecoration,就能实现不同的效果。

    二、RecycleView的优点

       1、支持局部刷新。
       2、可以自定义item增删时的动画。
       3、能够实现item拖拽和侧滑删除等功能。
       4、默认已实现View的复用,而且回收机制更加完善。

    三、RecycleView的使用方法

    首先要在build.gradle文件中添加引用

    compile 'com.android.support:recyclerview-v7:26.1.0'
    

    主页面布局:

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    
        <android.support.v7.widget.RecyclerView
            android:id="@+id/rv_list"
            android:layout_width="match_parent"
            android:layout_height="match_parent" />
    </LinearLayou>
    

    item布局:

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal">
    
        <TextView
            android:id="@+id/tv_content"
            android:layout_width="match_parent"
            android:layout_height="50dp"
            android:gravity="center"
            android:text="数据" />
    </LinearLayout>
    

    adapter代码:

    public class MyRecycleViewAdapter extends RecyclerView.Adapter<MyRecycleViewAdapter.MyHolder> {
    
        private List mList;//数据源
    
        MyRecycleViewAdapter(List list) {
            mList = list;
        }
    
        //创建ViewHolder并返回,后续item布局里控件都是从ViewHolder中取出
        @Override
        public MyHolder onCreateViewHolder(ViewGroup parent, int viewType) {
            //将我们自定义的item布局R.layout.item_one转换为View
            View view = LayoutInflater.from(parent.getContext())
                    .inflate(R.layout.item_one, parent, false);
            //将view传递给我们自定义的ViewHolder
            MyHolder holder = new MyHolder(view);
            //返回这个MyHolder实体
            return holder;
        }
        
        //通过方法提供的ViewHolder,将数据绑定到ViewHolder中
        @Override
        public void onBindViewHolder(MyHolder holder, int position) {
            holder.textView.setText(mList.get(position).toString());
        }
    
        //获取数据源总的条数
        @Override
        public int getItemCount() {
            return mList.size();
        }
    
        /**
         * 自定义的ViewHolder
         */
        class MyHolder extends RecyclerView.ViewHolder {
    
            TextView textView;
    
            public MyHolder(View itemView) {
                super(itemView);
                textView = itemView.findViewById(R.id.tv_content);
            }
        }
    }
    

    MainActivity代码:

    public class MainActivity extends AppCompatActivity {
    
        private RecyclerView mRecycleView;
        private MyRecycleViewAdapter mAdapter;//适配器
        private LinearLayoutManager mLinearLayoutManager;//布局管理器
        private List mList;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
    
            mList = new ArrayList();
            mRecycleView = findViewById(R.id.rv_list);
            //初始化数据
            initData(mList);
            //创建布局管理器,垂直设置LinearLayoutManager.VERTICAL,水平设置LinearLayoutManager.HORIZONTAL
            mLinearLayoutManager = new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false);
            //创建适配器,将数据传递给适配器
            mAdapter = new MyRecycleViewAdapter(mList);
            //设置布局管理器
            mRecycleView.setLayoutManager(mLinearLayoutManager);
            //设置适配器adapter
            mRecycleView.setAdapter(mAdapter);
        }
    
        public void initData(List list) {
            for (int i = 1; i <= 40; i++) {
                list.add("第" + i + "条数据");
            }
        }
    }
    

    Adapter

          使用时需要创建adapter(适配器)类,该类继承于RecyclerView.Adapter<VH>,其中VH是我们adapter类中创建的一个继承于RecyclerView.ViewHolder的静态内部类。

    可以看到该适配器类主要有3个方法和1个自定义ViewHolder组成:

    • onCreateViewHolder: 创建ViewHolder并返回,后续item布局里控件都是从ViewHolder中取出。
    • onBindViewHolder:通过方法提供的ViewHolder,将数据绑定到ViewHolder中。
    • getItemCount:获取数据源总的条数。
    • MyHolder :这是RecyclerView.ViewHolder的实现类,用于初始化item布局中的子控件。需要注意的是,在这个类的构造方法中需要传递item布局的View给父类 。

    使用方法:

    //设置适配器adapter
    mRecycleView.setAdapter(mAdapter);
    

    LayoutManager

          布局管理器,通过不同的布局管理器来控制item的排列顺序,负责item元素的布局和复用。RecycleView提供了三种布局管理器:

    • LinearLayoutManager:线性布局,以垂直或水平滚动列表方式显示项目。
    • GridLayoutManager:网格布局,在网格中显示项目。
    • StaggeredGridLayoutManager:瀑布流布局,在分散对齐网格中显示项目。

    使用方法:

    mRecycleView.setLayoutManager(new LinearLayoutManager(this,
                                      LinearLayoutManager.HORIZONTAL,false));
    

    运行效果:

    运行效果
          以上是LinearLayoutManager布局呈现的效果,假如遇到特殊需求,也可以通过继承RecyclerView.LayoutManager来自定义LayoutManager,重写它的方法来实现所需要的效果。

    ItemDecoration

          RecyclerView可以通过addItemDecoration()设置分割线。Android并没有提供实现好的分割线,所以任何的分割线样式都需要用户自己实现。可以通过继承RecyclerView.ItemDecoration类来实现。

    ItemDecoration源码:

    public abstract static class ItemDecoration {
    
            public void onDraw(Canvas c, RecyclerView parent, State state) {
                onDraw(c, parent);
            }
    
            @Deprecated
            public void onDraw(Canvas c, RecyclerView parent) {
            }
    
            public void onDrawOver(Canvas c, RecyclerView parent, State state) {
                onDrawOver(c, parent);
            }
    
            @Deprecated
            public void onDrawOver(Canvas c, RecyclerView parent) {
            }
    
            @Deprecated
            public void getItemOffsets(Rect outRect, int itemPosition, RecyclerView parent) {
                outRect.set(0, 0, 0, 0);
            }
    
            public void getItemOffsets(Rect outRect, View view, RecyclerView parent, State state) {
                getItemOffsets(outRect, ((LayoutParams) view.getLayoutParams()).getViewLayoutPosition(),
                        parent);
            }
        }
    

    该抽象类主要由三个方法组成:

    • onDraw(Canvas c, RecyclerView parent, State state):在Item绘制之前被调用(先于drawChildren),主要用于绘制分割线样式。
    • onDrawOver(Canvas c, RecyclerView parent, State state):在Item绘制之后被调用(慢于drawChildren),主要用于绘制分割线样式。
    • getItemOffsets(Rect outRect, View view, RecyclerView parent, State state):通过outRect.set()为每个Item设置一定的偏移量。

          onDrawonDrawOver这两个方法都是用于绘制分割线,我们在使用时只需要按需求选择一个进行实现就可以。它们的区别在于执行时间的不同,我们可以通过源码找到它们的区别。

    RecyclerView源码:

    public class RecyclerView extends ViewGroup implements ScrollingView, NestedScrollingChild2 {
        @Override
        public void draw(Canvas c) {
            super.draw(c);
    
            final int count = mItemDecorations.size();
            for (int i = 0; i < count; i++) {
                mItemDecorations.get(i).onDrawOver(c, this, mState);
            }
            
        @Override
        public void onDraw(Canvas c) {
            super.onDraw(c);
    
            final int count = mItemDecorations.size();
            for (int i = 0; i < count; i++) {
                mItemDecorations.get(i).onDraw(c, this, mState);
            }
        }
    }
    

    View源码:

    public class View implements Drawable.Callback, KeyEvent.Callback,
            AccessibilityEventSource {
             public void draw(Canvas canvas) {
                  // Step 3, draw the content
                  if (!dirtyOpaque) onDraw(canvas);
    
                  // Step 4, draw the children
                  dispatchDraw(canvas);
             }
    }
    

          首先执行的是RecyclerView中重写的draw()方法,然后会去执行super.draw(),即View的draw()方法。在View的draw()方法中,会先去执行onDraw(),再去执行dispatchDraw()方法,由于RecyclerView重写了onDraw()方法,所以是先执行了RecyclerView中的onDraw()方法。因此,它们的执行顺序为:onDraw()->dispatchDraw()->onDrawOver()。不理解的话可以参照上面的图多看两遍。

    Google给我们提供了一个实现类:DividerItemDecoration,我们可以参照它去实现自定义的Item Decoration。

    DividerItemDecoration源码:

    public class DividerItemDecoration extends RecyclerView.ItemDecoration {
        public static final int HORIZONTAL = LinearLayout.HORIZONTAL;
        public static final int VERTICAL = LinearLayout.VERTICAL;
    
        private static final String TAG = "DividerItem";
        private static final int[] ATTRS = new int[]{ android.R.attr.listDivider };
    
        private Drawable mDivider;
    
        private int mOrientation;
    
        private final Rect mBounds = new Rect();
    
        public DividerItemDecoration(Context context, int orientation) {
            final TypedArray a = context.obtainStyledAttributes(ATTRS);
            mDivider = a.getDrawable(0);
            if (mDivider == null) {
                Log.w(TAG, "@android:attr/listDivider was not set in the theme used for this "
                        + "DividerItemDecoration. Please set that attribute all call setDrawable()");
            }
            a.recycle();
            setOrientation(orientation);
        }
    
        public void setOrientation(int orientation) {
            if (orientation != HORIZONTAL && orientation != VERTICAL) {
                throw new IllegalArgumentException(
                        "Invalid orientation. It should be either HORIZONTAL or VERTICAL");
            }
            mOrientation = orientation;
        }
    
        public void setDrawable(@NonNull Drawable drawable) {
            if (drawable == null) {
                throw new IllegalArgumentException("Drawable cannot be null.");
            }
            mDivider = drawable;
        }
    
        @Override
        public void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state) {
            if (parent.getLayoutManager() == null || mDivider == null) {
                return;
            }
            if (mOrientation == VERTICAL) {
                drawVertical(c, parent);
            } else {
                drawHorizontal(c, parent);
            }
        }
    
        private void drawVertical(Canvas canvas, RecyclerView parent) {
            canvas.save();
            final int left;
            final int right;
            //noinspection AndroidLintNewApi - NewApi lint fails to handle overrides.
            if (parent.getClipToPadding()) {
                left = parent.getPaddingLeft();
                right = parent.getWidth() - parent.getPaddingRight();
                canvas.clipRect(left, parent.getPaddingTop(), right,
                        parent.getHeight() - parent.getPaddingBottom());
            } else {
                left = 0;
                right = parent.getWidth();
            }
    
            final int childCount = parent.getChildCount();
            for (int i = 0; i < childCount; i++) {
                final View child = parent.getChildAt(i);
                parent.getDecoratedBoundsWithMargins(child, mBounds);
                final int bottom = mBounds.bottom + Math.round(child.getTranslationY());
                final int top = bottom - mDivider.getIntrinsicHeight();
                mDivider.setBounds(left, top, right, bottom);
                mDivider.draw(canvas);
            }
            canvas.restore();
        }
    
        private void drawHorizontal(Canvas canvas, RecyclerView parent) {
            canvas.save();
            final int top;
            final int bottom;
            //noinspection AndroidLintNewApi - NewApi lint fails to handle overrides.
            if (parent.getClipToPadding()) {
                top = parent.getPaddingTop();
                bottom = parent.getHeight() - parent.getPaddingBottom();
                canvas.clipRect(parent.getPaddingLeft(), top,
                        parent.getWidth() - parent.getPaddingRight(), bottom);
            } else {
                top = 0;
                bottom = parent.getHeight();
            }
    
            final int childCount = parent.getChildCount();
            for (int i = 0; i < childCount; i++) {
                final View child = parent.getChildAt(i);
                parent.getLayoutManager().getDecoratedBoundsWithMargins(child, mBounds);
                final int right = mBounds.right + Math.round(child.getTranslationX());
                final int left = right - mDivider.getIntrinsicWidth();
                mDivider.setBounds(left, top, right, bottom);
                mDivider.draw(canvas);
            }
            canvas.restore();
        }
    
        @Override
        public void getItemOffsets(Rect outRect, View view, RecyclerView parent,
                RecyclerView.State state) {
            if (mDivider == null) {
                outRect.set(0, 0, 0, 0);
                return;
            }
            if (mOrientation == VERTICAL) {
                outRect.set(0, 0, 0, mDivider.getIntrinsicHeight());
            } else {
                outRect.set(0, 0, mDivider.getIntrinsicWidth(), 0);
            }
        }
    }
    

          从源码中可以看到该类是用系统中的android.R.attr.listDivider来作为分割线,通过DividerItemDecoration构造方法中的setOrientation(orientation)来设置分割线的方向。在getItemOffsets()中利用outRect.set()去设置了绘制的范围,再在onDraw()中进行真正的绘制。

    使用方法:

    //设置分割线
    mRecycleView.addItemDecoration(new DividerItemDecoration(this,                       
                                     DividerItemDecoration.VERTICAL));
    

    运行效果:


    带分割线的列表.png

    事件监听

        RecyclerView并没有给我们提供现成的点击事件监听,需要我们自己去实现。我们可以在RecyclerView的Adapter中自定义一个接口,并创建一个供其他类设置监听的方法。

    public class MyRecycleViewAdapter extends RecyclerView.Adapter<MyRecycleViewAdapter.MyHolder> {
    
        private List mList;//数据源
    
        private OnItemClickListener onItemClickListener;
    
        /**
         * 供外部调用设置监听
         * @param onItemClickListener
         */
        public void setOnItemClickListener(OnItemClickListener onItemClickListener) {
            this.onItemClickListener = onItemClickListener;
        }
    
        /**
         * 自定义的接口
         */
        public interface OnItemClickListener {
            void onItemClick(View view, int position);
        }
    
        //通过方法提供的ViewHolder,将数据绑定到ViewHolder中
        @Override
        public void onBindViewHolder(final MyHolder holder, int position) {
            holder.textView.setText(mList.get(position).toString());
            holder.itemView.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    if (onItemClickListener != null) {
                        onItemClickListener.onItemClick(v, holder.getAdapterPosition() + 1);
                    }
                }
            });
        }
    }
    

          以上省略了部分与该内容无关的代码。当我们定义好接口后,我们在onBindViewHolder()方法中为holder.itemView(itemView是列表中的每一个item项)设置了点击事件监听,然后在onClick()中判断是否有用户传递过onItemClickListener实例进来,有的话会调用他的onItemClick(),将点击事件转移到我们的自定义接口上,传给外面的调用者。调用者代码如下:

     mAdapter.setOnItemClickListener(new MyRecycleViewAdapter.OnItemClickListener() {
                @Override
                public void onItemClick(View view, int position) {
                    Toast.makeText(getApplicationContext(), "第" + position + "条数据", Toast.LENGTH_SHORT).show();
                }
            });
    

          到这里点击事件就完成了。如果你想实现长按也是同样的方法,在自定义的接口中多加一个长按的方法,然后holder.itemView调用setOnLongClickListener()去将长按事件转移到自定义的接口上。


    ItemAnimator 动画

          RecyclerView可以通过mRecyclerView.setItemAnimator(ItemAnimator animator)来设置添加和移除时的动画效果。ItemAnimator是一个抽象类,RecyclerView为我们提供了一个ItemAnimator的实现类DefaultItemAnimator

    使用方法:

    //设置动画效果
     mRecycleView.setItemAnimator(new DefaultItemAnimator());
    

          在adapter中添加两个方法,用于添加和移除Item。这里要注意的是,更新数据集要用notifyItemInserted(position)notifyItemRemoved(position) ,而不是notifyDataSetChanged(),否则没有动画效果。

        /**
         * 添加数据
         */
        public void addItem() {
            mList.add(0, "new ");
            notifyItemInserted(0);
        }
    
        /**
         * 移除数据
         * @param position
         */
        public void removeItem(int position) {
            mList.remove(position);
            notifyItemRemoved(position);
        }
    

    效果是按下底部“添加”按钮会在顶部插入数据,点击列表中的Item则删除该条数据。

    添加删除动画效果.gif
          如果我们对这种动画效果不满意,也可以去自定义各种动画效果。目前github上有许多开源的项目,例如RecyclerViewItemAnimators,我们可以直接去引用或学习它的动画效果。

    结论:以上就是RecyclerView的基本用法,看到这里可能很多人会觉得它比ListView复杂得多,很多东西都需要自己去定义,但正是这种定制性使得它具有良好的扩展性,我们可以根据具体需求去自定义自己想要实现的效果,相信你也会喜欢上这个控件!

    相关文章

      网友评论

          本文标题:Android控件RecycleView的基本用法

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