还在用ListView?

还在用Lisview?RecyclerView都已经出来一年多了!

想必大家多或多或少的接触过或者了解过RecyclerView,为什么没有用起来,原因大概如下?

  • ListView我用的挺好的,为什么要换RecyclerView?
  • ListView稳定,熟悉,还知道很多开源库,特别的好用!
  • RecyclerView不能添加头部,ListView能!

RecyclerView

RecyclerView最大的优势就是灵活,RecyclerView只需改变一行代码就可以变化多种不同的布局显示排版,这一点对于开发者是非常方便的!

还有RecyclerView.Adapter,比BaseAdapter做了更好的封装,把BaseAdapter的getView方法拆分成onCreateViewHolder方法和onBindViewHolder方法,强制需要创建ViewHolder,这样的好处就是避免了初学者写性能不佳的代码

在Andorid 5.0出来不久,我就已经写过RecyclerView的简单介绍以及基本使用,不了解的可以看看ListView升级版RecyclerView,了解过的同学可以忽略,并往下看。

在实战中我们会遇到什么问题?

get到下面的技能就能够在使用RcyclerView的大路上畅通无阻了!

  • 添加分割线
  • 添加点按效果
  • 列表动画
  • 改变某个数据保持当前位置
  • 添加头部尾部
  • 列表分组
  • 各种效果集成Demo
  • 更灵活的RecyclerView
添加分割线
//通过以下方法添加分割线
mRecyclerView.addItemDecoration(new DividerItemDecoration(this,
DividerItemDecoration.VERTICAL_LIST));

需要RecyclerView.ItemDecoration这个抽象类实现一些方法
(后续有更偷懒的方法)

/**
 * This class is from the v7 samples of the Android SDK. It's not by me!
 * <p/>
 * See the license above for details.
 */
public class DividerItemDecoration extends RecyclerView.ItemDecoration {

    private static final int[] ATTRS = new int[]{
            android.R.attr.listDivider
    };

    public static final int HORIZONTAL_LIST = LinearLayoutManager.HORIZONTAL;

    public static final int VERTICAL_LIST = LinearLayoutManager.VERTICAL;

    private Drawable mDivider;

    private int mOrientation;

    public DividerItemDecoration(Context context, int orientation) {
        final TypedArray a = context.obtainStyledAttributes(ATTRS);
        mDivider = a.getDrawable(0);
        a.recycle();
        setOrientation(orientation);
    }

    public void setOrientation(int orientation) {
        if (orientation != HORIZONTAL_LIST && orientation != VERTICAL_LIST) {
            throw new IllegalArgumentException("invalid orientation");
        }
        mOrientation = orientation;
    }

    @Override
    public void onDraw(Canvas c, RecyclerView parent) {
        Log.v("recyclerview - itemdecoration", "onDraw()");

        if (mOrientation == VERTICAL_LIST) {
            drawVertical(c, parent);
        } else {
            drawHorizontal(c, parent);
        }

    }


    public void drawVertical(Canvas c, RecyclerView parent) {
        final int left = parent.getPaddingLeft();
        final int right = parent.getWidth() - parent.getPaddingRight();

        final int childCount = parent.getChildCount();
        for (int i = 0; i < childCount; i++) {
            final View child = parent.getChildAt(i);
            android.support.v7.widget.RecyclerView v = new android.support.v7.widget.RecyclerView(parent.getContext());
            final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child
                    .getLayoutParams();
            final int top = child.getBottom() + params.bottomMargin;
            final int bottom = top + mDivider.getIntrinsicHeight();
            mDivider.setBounds(left, top, right, bottom);
            mDivider.draw(c);
        }
    }

    public void drawHorizontal(Canvas c, RecyclerView parent) {
        final int top = parent.getPaddingTop();
        final int bottom = parent.getHeight() - parent.getPaddingBottom();

        final int childCount = parent.getChildCount();
        for (int i = 0; i < childCount; i++) {
            final View child = parent.getChildAt(i);
            final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child
                    .getLayoutParams();
            final int left = child.getRight() + params.rightMargin;
            final int right = left + mDivider.getIntrinsicHeight();
            mDivider.setBounds(left, top, right, bottom);
            mDivider.draw(c);
        }
    }

    @Override
    public void getItemOffsets(Rect outRect, int itemPosition, RecyclerView parent) {
        if (mOrientation == VERTICAL_LIST) {
            outRect.set(0, 0, 0, mDivider.getIntrinsicHeight());
        } else {
            outRect.set(0, 0, mDivider.getIntrinsicWidth(), 0);
        }
    }
}

看到这里应该有同学吐槽了,就添加一根线至于吗?还需要重写方法,ListView可是只需要简单配置一下就好了。

   android:divider="@color/grey"
   android:dividerHeight="5dp"

为此我也感觉太麻烦,于是我想了一个办法:
直接在item_view里面底部自己添加一根线布局,这样就无需重写了,并且这样还有个好处就是,如果细心的同学会发现,添加分割线,最后一个item下面不会有分割线,显然当数据量不足一个屏幕的时候显得很突兀,但是在item_view下面添加一个线的布局则不会出现这种情况

添加点按效果

RecyclerView直接在item_view里面配置即可

动画

一个好的用户体验就是要有操作动画的过渡,而不是生硬的刷新列表。

推荐一个RecyclerView的动画库(recyclerview-animators
这里写图片描述

RecyclerView自带添加、删除动画,而ListView则需添加额外的代码才能实现。

  • 删除调用RecyclerView的adapter的notifyItemRemoved
  • 添加调用RecyclerView的adapter的notifyItemInserted

说到adapter我们就来说说RecyclerView.Adapter和BaseAdapter相比,额外提供了一下这些方法:

// 数据发生了改变,那调用这个方法,传入改变对象的位置。
public final void notifyItemChanged(int position);

// 可以刷新从positionStart开始itemCount数量的item了
public final void notifyItemRangeChanged(int positionStart, int itemCount);

// 添加,传入对象的位置。
public final void notifyItemInserted(int position);

// 删除,传入对象的位置。
public final void notifyItemRemoved(int position);

// 对象从fromPosition移动到toPosition
 public final void notifyItemMoved(int fromPosition, int toPosition); 

//批量添加 
public final void notifyItemRangeInserted(int positionStart, int itemCount);

//批量删除
public final void notifyItemRangeRemoved(int positionStart, int itemCount);
改变列表某个布局状态且保持当前位置

这种需求是普遍存在的,就是改变列表某一个item数据,然后刷新列表,如果是ListView刷新后则会回到最顶部,而RecyclerView同样的操作但是原来滑动的位置不变。

各种解决方案的RecyclerView的Adapter

BaseRecyclerViewAdapterHelper

效果展示

更灵活的RecyclerView

twoway-view
封装了RecyclerView常用方法,如click等等,以及支持了更多不同的布局,使得RecyclerView使用起来更简单!
这里写图片描述

造起来!小伙伴们!

ListView 是 Android 常用的一种控件,用于展示一系列有序的项目,通常是从数据库或数据源获取的数据。使用 ListView 的基本步骤如下: 1. **添加到布局文件**: 在 XML 布局文件,添加一个 `ListView` 元素,并设置其 ID 和必要的属性,如宽度、高度等。 ```xml <ListView android:id="@+id/list_view" android:layout_width="match_parent" android:layout_height="wrap_content" android:scrollbars="vertical" /> ``` 2. **创建适配器**: 创建一个 `ArrayAdapter` 或者自定义的 `BaseAdapter` 类,负责将数据源转换为 `ListView` 可显示的视图。你需要重写 `getView()` 方法来生成列表项。 ```java ArrayAdapter<String> adapter = new ArrayAdapter<>(context, layoutResourceId, items); list_view.setAdapter(adapter); ``` 3. **数据源**: 数据可以来自静态数组、动态生成的数据(如从网络请求)、SQLite 数据库等。确保适配器的 `notifyDataSetChanged()` 方法在数据源改变后被调用,以更新列表。 4. **监听事件**: 可以给 `ListView` 设置点击事件监听器,当用户点击列表项时,执行相应的操作。 ```java list_view.setOnItemClickListener(new AdapterView.OnItemClickListener() { @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id) { // 处理点击事件 } }); ``` 5. **设置分隔符和选择模式**: 如果需要,可以设置 `ListView` 的分隔符样式和选择模式。 ```java list_view.setDividerDrawable Resources.getDrawable(R.drawable.my_divider)); list_view.setChoiceMode(ListView.CHOICE_MODE_SINGLE); ```
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值