Android RecyclerView多类型布局卡片解决方案

背景

随着公司业务越来越复杂,在同一个列表中需要展示各种类型的数据。

总体结构

  • ItemViewAdapter: 每种类型的卡片分别都是不同的ItemViewAdapter
  • ItemViewAdapterFactory: 使用ItemViewAdapterFactory根据不同数据对应不同的ItemViewAdapter
  • MultiRecyclerViewAdapter: MultiRecyclerViewAdapter就是RecylerView.Adapter,并是个ItemViewAdapterFactory。 具体只要继承MultiRecyclerViewAdapter即可,实现ItemViewAdapterFactory中getViewType、onCreateItemViewAdapter两个方法
  • ContextMap: 整个Adapter共用一个ContextMap数据上下文,用于外部(例Fragment等)与ItemAdapter交互、ItemAdapter之间交互等一系列数据传递,可以解决参数层层传递的问题
  • RecyclerViewHolder: 通用RecyclerView.ViewHolder,封装根据id获取view方法getView(viewId)、获取数据上下文方法getContextMap()

使用方法

  • 每种类型卡片Item都实现ItemViewAdapter
package com.lkh.multiadapter;

import android.support.annotation.LayoutRes;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import java.util.List;

/**
 * 列表单项布局与数据绑定
 * Created by luokanghui on 2017/5/24.
 */
public abstract class ItemViewAdapter<E>{

    /**
     * 返回列表单项View,如果View由资源layout加载而来,直接重写{@link #onGetLayoutId()}即可
     * @param parent 父view,一般为RecyclerView
     * @return 列表单项View
     */
    public View onCreateView(ViewGroup parent){
        return LayoutInflater.from(parent.getContext()).inflate(onGetLayoutId()
                , parent, false);
    }

    /**
     * 当RecyclerViewHolder创建成功后调用,只会调用一次
     * @param viewHolder 单项view集合
     */
    public void onCreate(RecyclerViewHolder viewHolder){

    }

    /**
     * 返回单项布局的资源id,如果重写了{@link #onCreateView(ViewGroup)},则此方法可能失效
     * @return 单项布局layout id
     */
    @LayoutRes
    protected abstract int onGetLayoutId();

    /**
     * 把数据与view进行绑定,滑动时都会调用
     * @param viewHolder 单项view集合
     * @param data 具体数据
     * @param position 在列表中的位置
     */
    public abstract void bindData(RecyclerViewHolder viewHolder, E data, int position);

    /**
     * 局部更新时调用
     * @param viewHolder 单项view集合
     * @param data 具体数据
     * @param position 在列表中的位置
     * @param payloads 局部更新标志,不会为空(isEmpty()==false)
     */
    public void bindData(RecyclerViewHolder viewHolder, E data, int position, List<Object> payloads){

    }
}
复制代码

卡片1:

package com.lkh.multiadapter.sample;

import android.widget.TextView;

import com.lkh.multiadapter.ItemViewAdapter;
import com.lkh.multiadapter.R;
import com.lkh.multiadapter.RecyclerViewHolder;

/**
 * 卡片1实现
 * Created by luokanghui on 2019/3/18
 */
public class SampleOneItemViewAdapter extends ItemViewAdapter<DataOne> {
    @Override
    protected int onGetLayoutId() {
        //布局layout资源id
        return R.layout.item_one;
    }

    @Override
    public void bindData(RecyclerViewHolder viewHolder, DataOne data, int position) {
        //根据id获取view
        TextView tvContent = viewHolder.getView(R.id.tv_content);
        //数据绑定
        tvContent.setText(data.getContent());
    }
}
复制代码
package com.lkh.multiadapter.sample;

/**
 * 卡片1数据
 * Created by luokanghui on 2019/3/18
 */
public class DataOne {
    private String content;

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }
}
复制代码

item_one.xml:

<?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:background="#eeeeee"
    android:orientation="vertical">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="卡片1"
        android:textColor="#000000" />

    <TextView
        android:id="@+id/tv_content"
        android:layout_width="wrap_content"
        android:layout_height="50dp"
        android:gravity="center"
        android:textColor="#000000" />

</LinearLayout>
复制代码

卡片2

package com.lkh.multiadapter.sample;

import android.widget.TextView;

import com.lkh.multiadapter.ItemViewAdapter;
import com.lkh.multiadapter.R;
import com.lkh.multiadapter.RecyclerViewHolder;

/**
 * 卡片2实现
 * Created by luokanghui on 2019/3/18
 */
public class SampleTwoItemViewAdapter extends ItemViewAdapter<DataTwo> {
    @Override
    protected int onGetLayoutId() {
        //布局layout资源id
        return R.layout.item_two;
    }

    @Override
    public void bindData(RecyclerViewHolder viewHolder, DataTwo data, int position) {
        //根据id获取view
        TextView tvNum = viewHolder.getView(R.id.tv_num);
        //数据绑定
        tvNum.setText("num="+data.getNum());
    }
}
复制代码
package com.lkh.multiadapter.sample;

/**
 * 卡片2数据
 * Created by luokanghui on 2019/3/18
 */
public class DataTwo {
    private int num;

    public int getNum() {
        return num;
    }

    public void setNum(int num) {
        this.num = num;
    }
}
复制代码

item_two.xml:

<?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:background="#999999"
    android:orientation="vertical">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="卡片2"
        android:textColor="#0000ff" />

    <TextView
        android:id="@+id/tv_num"
        android:layout_width="wrap_content"
        android:layout_height="100dp"
        android:gravity="center"
        android:textColor="#0000ff" />

</LinearLayout>
复制代码
  • 总Adapter,继承MultiRecyclerViewAdapter
package com.lkh.multiadapter.sample;

import com.lkh.multiadapter.ItemViewAdapter;
import com.lkh.multiadapter.MultiRecyclerViewAdapter;

/**
 * 多布局adapter,根据不同data及position,使用不同ItemViewAdapter卡片
 * Created by luokanghui on 2019/3/18
 */
public class SampleMultiAdapter extends MultiRecyclerViewAdapter<Object> {
    private static final int TYPE_EMPTY = 0;//空item
    private static final int TYPE_ONE = 1;//卡片1
    private static final int TYPE_TWO = 2;//卡片2

    @Override
    public int getViewType(Object data, int position) {
        if (data instanceof DataOne){//卡片1
            return TYPE_ONE;
        }

        if (data instanceof DataTwo){//卡片2
            return TYPE_TWO;
        }

        return TYPE_EMPTY;//空item
    }

    @Override
    public ItemViewAdapter onCreateItemViewAdapter(int viewType) {
        switch (viewType){
            case TYPE_ONE://卡片1
                return new SampleOneItemViewAdapter();
            case TYPE_TWO://卡片2
                return new SampleTwoItemViewAdapter();
            default://空item
                return new EmptyItemViewAdapter();
        }
    }
}
复制代码
  • RecyclerView中使用
package com.lkh.multiadapter;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;

import com.lkh.multiadapter.sample.DataOne;
import com.lkh.multiadapter.sample.DataTwo;
import com.lkh.multiadapter.sample.SampleMultiAdapter;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        RecyclerView recyclerView = findViewById(R.id.recycler_view);
        recyclerView.setLayoutManager(new LinearLayoutManager(this));

        SampleMultiAdapter adapter = new SampleMultiAdapter();

        //设置数据
        adapter.setData(generateData());

        //设置adapter
        recyclerView.setAdapter(adapter);

    }

    //造测试数据
    private List<Object> generateData(){
        List<Object> list = new ArrayList<>();
        for (int i=0; i<20; i++){
            DataOne dataOne = new DataOne();
            dataOne.setContent("这是卡片1数据:"+i);
            list.add(dataOne);

            DataTwo dataTwo = new DataTwo();
            dataTwo.setNum(i);
            list.add(dataTwo);
        }
        return list;
    }
}
复制代码

activity_main.xml:

<?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="match_parent"
    android:orientation="vertical">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/recycler_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>

</LinearLayout>
复制代码
  • 运行效果如下:

    )

  • 总的来说,实现一个多类型布局列表只需要写多个不同卡片ItemViewAdapter、继承MultiRecyclerViewAdapter用来控制不同数据使用不同ItemViewAdapter,新增一个卡片只需要新增一个ItemViewAdapter,在MultiRecyclerViewAdapter新加一项即可,不会影响其它卡片使用,而且ItemViewAdapter完全独立,可以很好的复用。

核心代码

package com.lkh.multiadapter;

import android.support.v7.widget.RecyclerView;
import android.view.ViewGroup;

import java.util.List;

/**
 * 多种布局adapter
 * Created by luokanghui on 2017/5/24.
 */

public abstract class MultiRecyclerViewAdapter<E> extends RecyclerView.Adapter<RecyclerViewHolder> implements ItemViewAdapterFactory<E> {
    public static final int NO_TYPE = -1;

    private List<E> dataList;

    protected final MapData mMapData = new MapData();

    public MultiRecyclerViewAdapter setData(List<E> list) {
        this.dataList = list;
        return this;
    }

    @Override
    public RecyclerViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        ItemViewAdapter itemViewModule = onCreateItemViewAdapter(viewType);
        RecyclerViewHolder recyclerViewHolder = new RecyclerViewHolder(itemViewModule.onCreateView(parent), itemViewModule, this, getContextMap());
        itemViewModule.onCreate(recyclerViewHolder);
        return recyclerViewHolder;
    }

    @Override
    public void onBindViewHolder(RecyclerViewHolder holder, int position) {
        if (checkItems(position)) {
            return;
        }
        holder.itemViewAdapter.bindData(holder, dataList.get(position), position);
    }

    @Override
    public void onBindViewHolder(RecyclerViewHolder holder, int position, List<Object> payloads) {
        if (checkItems(position)) {
            return;
        }
        if (payloads.isEmpty()) {
            super.onBindViewHolder(holder, position, payloads);
        } else {
            holder.itemViewAdapter.bindData(holder, dataList.get(position), position, payloads);
        }
    }

    @Override
    public int getItemViewType(int position) {
        if (checkItems(position)) {
            return NO_TYPE;
        }
        return getViewType(dataList.get(position), position);
    }

    @Override
    public int getItemCount() {
        return dataList == null ? 0 : dataList.size();
    }

    /**
     * true表示没通过
     */
    private boolean checkItems(int position) {
        return dataList == null || position < 0 || position >= dataList.size();
    }

    @Override
    public MapData getContextMap() {
        return mMapData;
    }
}
复制代码
package com.lkh.multiadapter;


/**
 * 多布局ItemViewAdapter创建者
 * Created by luokanghui on 2017/5/24.
 */
public interface ItemViewAdapterFactory<E> {

    /**
     * 返回ItemViewAdapter的类型
     * 建议根据data的数据类型判断不同的viewType
     * @param data 具体数据
     * @param position 在列表中的位置
     * @return 类型
     */
    int getViewType(E data, int position);

    /**
     * 根据不同的viewType返回不同的ItemViewAdapter
     * @param viewType 类型
     * @return ItemViewAdapter
     */
    ItemViewAdapter<? extends E> onCreateItemViewAdapter(int viewType);


    /**
     * 上下文数据
     * @return
     */
    MapData getContextMap();
}
复制代码
package com.lkh.multiadapter;

import android.support.v7.widget.RecyclerView;
import android.util.SparseArray;
import android.view.View;


/**
 * ViewHolder基类
 */
public final class RecyclerViewHolder extends RecyclerView.ViewHolder {

    private final SparseArray<View> views;
    ItemViewAdapter itemViewAdapter;
    private final RecyclerView.Adapter adapter;
    private final MapData mMapData ;


    public RecyclerViewHolder(View itemView, ItemViewAdapter itemViewAdapter, RecyclerView.Adapter adapter, MapData mapData) {
        super(itemView);
        this.views = new SparseArray<>();
        this.itemViewAdapter = itemViewAdapter;
        this.adapter = adapter;
        this.mMapData = mapData;
    }

    /**
     * 根据id获取view,如果缓存中存在,直接使用缓存中的,避免重复执行findViewById
     */
    @SuppressWarnings("unchecked")
    public <T extends View> T getView(int viewId) {
        View view = views.get(viewId);
        if (view == null) {
            view = itemView.findViewById(viewId);
            views.put(viewId, view);
        }
        return (T) view;
    }

    public RecyclerView.Adapter getAdapter(){
        return adapter;
    }

    /**
     * 获取数据上下文
     */
    public MapData getContextMap(){
        return mMapData;
    }


}
复制代码
package com.lkh.multiadapter;

import android.support.annotation.LayoutRes;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

import java.util.List;

/**
 * 列表单项布局与数据绑定
 * Created by luokanghui on 2017/5/24.
 */

public abstract class ItemViewAdapter<E>{

    /**
     * 返回列表单项View,如果View由资源layout加载而来,直接重写{@link #onGetLayoutId()}即可
     * @param parent 父view,一般为RecyclerView
     * @return 列表单项View
     */
    public View onCreateView(ViewGroup parent){
        return LayoutInflater.from(parent.getContext()).inflate(onGetLayoutId()
                , parent, false);
    }

    /**
     * 当RecyclerViewHolder创建成功后调用,只会调用一次
     * @param viewHolder 单项view集合
     */
    public void onCreate(RecyclerViewHolder viewHolder){

    }

    /**
     * 返回单项布局的资源id,如果重写了{@link #onCreateView(ViewGroup)},则此方法可能失效
     * @return 单项布局layout id
     */
    @LayoutRes
    protected abstract int onGetLayoutId();

    /**
     * 把数据与view进行绑定,滑动时都会调用
     * @param viewHolder 单项view集合
     * @param data 具体数据
     * @param position 在列表中的位置
     */
    public abstract void bindData(RecyclerViewHolder viewHolder, E data, int position);

    /**
     * 局部更新时调用
     * @param viewHolder 单项view集合
     * @param data 具体数据
     * @param position 在列表中的位置
     * @param payloads 局部更新标志,不会为空(isEmpty()==false)
     */
    public void bindData(RecyclerViewHolder viewHolder, E data, int position, List<Object> payloads){

    }
}
复制代码
  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: RecyclerView 是一个用于在 Android 应用中显示大量数据列表的强大工具。它是 ListView 和 GridView 的改进版本,具有更高的灵活性和性能优化。 RecyclerView 采用了 ViewHolder 模式来管理列表项视图的缓存,以便在滚动时提高性能。此外,RecyclerView 还支持多种布局管理器,可以轻松地实现不同的列表布局,例如线性布局、网格布局和瀑布流布局等。 要在 Android 应用中使用 RecyclerView,需要完成以下步骤: 1. 添加 RecyclerView 依赖库到应用的 build.gradle 文件中: ``` implementation 'androidx.recyclerview:recyclerview:1.2.1' ``` 2. 在布局文件中添加 RecyclerView 控件: ``` <androidx.recyclerview.widget.RecyclerView android:id="@+id/recyclerView" android:layout_width="match_parent" android:layout_height="match_parent" /> ``` 3. 创建 RecyclerView 的 Adapter 类,并实现其中的必要方法: ``` class MyAdapter(private val dataList: List<String>) : RecyclerView.Adapter<MyAdapter.MyViewHolder>() { override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): MyViewHolder { val view = LayoutInflater.from(parent.context) .inflate(R.layout.item_layout, parent, false) return MyViewHolder(view) } override fun onBindViewHolder(holder: MyViewHolder, position: Int) { holder.bind(dataList[position]) } override fun getItemCount() = dataList.size class MyViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) { fun bind(item: String) { itemView.findViewById<TextView>(R.id.textView).text = item } } } ``` 4. 在 Activity 或 Fragment 中设置 RecyclerView 的 LayoutManager 和 Adapter: ``` val recyclerView = findViewById<RecyclerView>(R.id.recyclerView) val layoutManager = LinearLayoutManager(this) val adapter = MyAdapter(listOf("item 1", "item 2", "item 3")) recyclerView.layoutManager = layoutManager recyclerView.adapter = adapter ``` 以上就是使用 RecyclerView 实现多布局的基本步骤。可以根据需要进行进一步的自定义和优化。 ### 回答2: Android RecyclerView 是一款强大的控件,它让我们可以轻松创建与管理数据集合,以及快速显示它们。其中,在RecyclerView 中使用多布局是非常常见的,特别是在需要展示多种类型的数据时。在这种情况下,使用多布局可以让我们根据每种数据类型的不同,选择正确的布局以呈现更好的用户体验。 要在 RecyclerView 中使用多布局,我们需要进行以下步骤: 1.创建一组不同的布局,以满足不同的数据类型。我们可以在布局文件中定义不同的 RecyclerView ViewHolder,每个 ViewHolder 都对应着不同类型的数据。 2.在 Adapter 中实现 getItemViewType() 方法,该方法用于确定每个 ViewHolder 对应的数据类型。我们可以根据数据内容,选择不同的布局类型,比如可以使用 item 类型表示某个具体的数据,使用 header 类型表示一些相关的信息。 3.在 Adapter 中实现 onCreateViewHolder() 方法,该方法根据 View type 创建不同的 ViewHolder。我们可以通过 LayoutInflater API 将不同类型布局加载到新创建的 ViewHolder 中。 4.在 onBindViewHolder() 中,我们可以根据 ViewHolder position 来确定该 ViewHolder 显示的数据,为 ViewHolder 中的视图设置内容。 因此,以上步骤可以帮助我们在 RecyclerView 中实现多布局。需要注意的是,在使用多布局时,我们需要特别小心,确保不要因为过多的复杂逻辑而影响 RecyclerView 的性能。 ### 回答3: Android RecyclerView是一个高级版本的ListView,它提供了更多的灵活性和可定制性。RecyclerView支持很多种布局格式,其中一种便是多布局。所谓“多布局”,是指在RecyclerView中,不同位置的item使用不同的布局文件。需要在Adapter中重写getItemViewType()方法,返回不同的viewType,这样就可以根据不同的viewType来选择不同的布局文件。 多布局的应用场景常见且广泛,比如一个聊天窗口,左侧为接受的消息,右侧为发送的消息,这就是两种不同的布局;再比如一个社交软件中的朋友圈,其中有纯文字、文字加图片、文字加视频等多种不同的布局方式。 在RecyclerView的Adapter中,可以继承RecyclerView.Adapter<T>,并重写其中的几种方法。为了支持多种布局,主要需要重写getItemViewType和onCreateViewHolder方法。 getItemViewType方法用于获取当前item的类型。需要根据position的位置,判断需要使用哪种类型布局文件。在方法中,可以使用一个switch-case去判断,并根据不同的position来返回对应的类型。它需要返回一个int值,表示当前item的类型。 onCreateViewHolder方法用于创建ViewHolder对象。ViewHolder对象用来绑定视图控件,使其能够正确显示数据。在这个方法中,可以根据item类型去创建不同的ViewHolder并返回。同时,需要注意,该方法中的layoutInflate.inflate()方法需要根据viewType参数来加载不同的布局文件。 另外,需要在Adapter中重写getItemCount()方法,返回item的数量,以确保RecyclerView中正确显示所有的item。 综上所述,RecyclerView之所以受到开发者的喜爱,是因为其强大高效的布局管理和灵活的定制能力。通过实现多布局,可以实现各种不同的布局方式,提高用户体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值