Android中对RecyclerView Adapter封装解析
前言
关于adapter的封装,网上有很多开源库,开发的时候可以直接拿来用,省了很多事。
最近闲来无事,想着自己动手封装一个adapter。
问题
1、通常我们封装的时候,可以简化到这一步:
baserecyclerviewadapter adapter = new baserecyclerviewadapter() { private static final int type_fir = 1; private static final int type_sec = 2; private static final int type_thr = 3; @override public int getlayoutid(int viewtype) { if (viewtype == type_fir) { return r.layout.test_activity_recyclerview_item; } else if (viewtype == type_sec){ return r.layout.test_activity_recyclerview_item_two; } else { return r.layout.test_activity_recyclerview_item_three; } } @override public int getitemviewtype(int position) { if (position % 3 == 1) { return type_fir; } else if (position % 3 == 2) { return type_sec; } else { return type_thr; } } @override public void onbindrecyclerviewholder(baseviewholder holder, int position) { if (getitemviewtype(position) == type_fir) { // todo 数据处理及绑定 } else if (getitemviewtype(position) == type_sec) { // todo 数据处理及绑定 } else { // todo 数据处理及绑定 } } };
2、从上面代码我们可以看到,当处理多布局类型的时候,我们需要解决的是:
- 多布局类型定义
- 返回什么类型处理
- 根据指定类型,加载对应布局layout
- 根据类型,处理及绑定数据
3、当我们迭代更新的时候,我们只能通过修改adapter,这样耦合性稍微有点强。
不过呢,简化到这一步的时候,其实在项目中是可以使用的
解决
1、上面的问题,我们知道,如果需要迭代更新,我们只能通过修改viewtype、layout、onbind等具体数据来实现。既然这样,我们可以将这些具体数据抽取封装起来,adapter不直接处理数据,而是交给这个封装类,逻辑交给封装类来处理,这样adapter只需要负责维护这个封装类即可。
2、这个封装类,我们暂且称作item
interface item { @layoutres int getlayoutresource(); int getitemviewtype(); baseviewholder oncreateviewholder(viewgroup parent, int viewtype); void onbindviewholder(baseviewholder holder, int position); }
adpater中,需要动态调整的就是
- viewtype
- layout布局
- oncreateviewholder
- onbindviewholder
所以就将这些方法抽取封装起来,然后adapter维护item列表即可
3、上面这个接口item
第一:没有数据,我们需要自己在实现类中提供数据设置获取方法;
第二:我们需要同时实现四个方法
既然这样,我们可以通过一个基类baseitem,来简化操作
4、在item接口中,只有oncreateviewholder()方法,是不需要每次都实现的,因为它是固定不变的的,baseitem就可以这样写:
public abstract class baseitem<t> implements item { public t mdata; public void setdata(t t) { this.mdata = t; } @override public baseviewholder oncreateviewholder(viewgroup parent, int viewtype) { int resourceid = getlayoutresource(); view itemview = layoutinflater.from(parent.getcontext()).inflate(resourceid, parent, false); return new baseviewholder(itemview); } }
5、现在,我们只需要实现三个方法就可以了。我们再看getitemviewtype,这个返回的是当前布局类型,int类型,唯一且不能重复 。android中的layout维护着一个int型的唯一标识id,我们是不是可以拿这个标识id,作为viewtype?这样就可以进一步简化实现类,现在我们只需实现两个方法即可。
public abstract class baseitem<t> implements item { public t mdata; public void setdata(t t) { this.mdata = t; } @override public int getitemviewtype() { return getlayoutresource(); } @override public baseviewholder oncreateviewholder(viewgroup parent, int viewtype) { int resourceid = getlayoutresource(); view itemview = layoutinflater.from(parent.getcontext()).inflate(resourceid, parent, false); return new baseviewholder(itemview); } }
6、接下来,我们要构造adapter,使之不再维护具体数据,而是baseitem列表
public class baseadapter extends recyclerview.adapter<baseviewholder> { private list<baseitem> mdata; public baserecyclerviewadapter(list<baseitem> data) { mdata = new arraylist<>(); if (data != null) { mdata.addall(data); } } @override public baseviewholder oncreateviewholder(viewgroup parent, final int viewtype) { for (baseitem item : mdata) { if (viewtype == item.getitemviewtype()) { return item.oncreateviewholder(parent, viewtype);; } } } @override public void onbindviewholder(baseviewholder holder, int position) { mdata.get(dataposition).onbindviewholder(holder, position); } @override public int getitemcount() { return mdata.size(); } @override public int getitemviewtype(int position) { return mdata.get(position).getitemviewtype(); } }
结语
以上简单介绍了一下封装思路,至于像头布局、脚布局、空布局、加载更多布局等等,其实都是adapter中的一种viettype布局,具体实现代码,github:baseadapter
下图是一adapter中数据构造:
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。