通用适配器(一)── 适用于 ListView、GridView 的 CommonBaseAdapte(ViewHolder 模式)

一、前言

项目中经常用到ListView、GridView。因此写了很多 Adapter。发现其中的代码非常类似。仅有的区别在于 viewHolder 的创建、将数据绑定到 viewHolder。所以就想能不能偷个懒。

二、思路

1、把不变的代码放到一个类里面,起名 CommonBaseAdapter。(它的构造器以自定义的 CommonViewHolder 的类(.class)为参数。在 getView 函数中,利用反射,实例化一个 viewHolder。之后,利用多态,绑定数据。)

2、把可变的代码抽象一下,创建一个虚基类,起名 CommonViewHolder。

3、需要一个新的 Adapter 的时候,继承CommonViewHolder,重写里面的函数。(这里要写的代码的全部是自己的业务逻辑。比如,在构造器里调用 convertView.findViewById 初始化自己的域、在 setItem 里绑定数据或者设置监听器等等。)

4、以自定义的 CommonViewHolder 的类(.class)为参数创建出可用于ListView、GridView 的 adapter。

三、源码

/**
 * @param <T>
 */
public class CommonBaseAdapter<T> extends BaseAdapter {
    protected static final String TAG = CommonBaseAdapter.class.getSimpleName();;

    private Context context;
    private int layout;
    private Class<? extends CommonViewHolder<T>> viewHolderClass;
    private List<T> datas;


    /**
     * @param context
     * @param layout
     * @param datas
     * @param viewHolderClass
     */
    public CommonBaseAdapter(Context context, int layout, List<T> datas, Class<? extends CommonViewHolder<T>> viewHolderClass) {
        super();
        this.context = context;
        this.layout = layout;
        this.datas = datas;
        this.viewHolderClass = viewHolderClass;
    }

    /**
     * @param position
     * @param convertView
     * @param parent
     * @return
     */
    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        CommonViewHolder<T> holder = null;
        if (convertView == null) {
            convertView = LayoutInflater.from(context).inflate(layout, parent, false);
            try {
                holder = viewHolderClass.getDeclaredConstructor(View.class).newInstance(convertView);
            } catch (NoSuchMethodException e){
                e.printStackTrace();
                Log.e(TAG, e.toString());
            } catch (SecurityException e){
                e.printStackTrace();
                Log.e(TAG, e.toString());
            } catch (Exception e) {
                e.printStackTrace();
                Log.e(TAG, e.toString());
            }
            convertView.setTag(holder);
        } else {
            holder = (CommonViewHolder<T>) convertView.getTag();
        }
        holder.setItem(datas.get(position));
        return convertView;
    }

    /**
     * @return
     */
    @Override
    public int getCount() {
        return datas.size();
    }

    /**
     * @param position
     * @return
     */
    @Override
    public Object getItem(int position) {
        return null;
    }

    /**
     * @param position
     * @return
     */
    @Override
    public long getItemId(int position) {
        return 0;
    }

    /**
     * @param <T>
     */
    public static abstract class CommonViewHolder<T> {
        /**
         * @param item
         */
        public abstract void setItem(T item);
    }
}

四、Sample

adapter = new CommonBaseAdapter<Image>(context, R.layout.grid_item_similar_img, datas, SimilarImgGridViewHolder.class);
similarImgGridView.setAdapter(adapter);

    static class SimilarImgGridViewHolder extends CommonBaseAdapter.CommonViewHolder<Image> {
        private ImageView imageView;

        public SimilarImgGridViewHolder(View convertView) {
            ((CardView)convertView).setPreventCornerOverlap(false);
            imageView = (ImageView) convertView.findViewById(R.id.iv_similar_img_grid_item);
        }

        @Override
        public void setItem(Image item) {
            ImageLoader.getInstance().displayImage(item.getDownloadUrl(), imageView);
        }
    }


五、弯路

因为 CommonViewHolder 只有 CommonBaseAdapter 才会用到,所以想把前者写成后者的内部类。然后问题就来了。利用反射实例化自定义的CommonViewHolder 的时候总是 Null。然后看文档,看到这句话,**If this Class object represents an inner class declared in a non-static context, the formal parameter types include the explicit enclosing instance as the first parameter.**然后就明白了,如果没有把 CommonViewHolder修饰为 static 的话,它的构造器会有个隐性参数—CommonBaseAdapter。(很好理解,非静态内部类会持有外部类的引用嘛)。这样的话,利用反射查找参数为 View的构造器的时候,就找不到,自然也就无法实例化。把CommonViewHolder修饰为 static解决问题。

通过封装BaseAdapter和RecyclerView.Adapter得到的通用的,简易的Adapter。项目地址:https://github.com/tianzhijiexian/CommonAdapter 效果图:已解决的问题 提升item的独立性,完美支持item被多处复用 item会根据type来做自动复用 支持多种类型的item 一个item仅会调用一次setViews(),避免重复建立监听器 一个item仅会触发一次绑定视图的操作,提示效率 支持dataBinding和其他第三方注入框架 提供了getView()方法来简化findViewById 支持通过item的构造方法来传入Activity对象 支持通过item的构造方法来传入item中事件的回调 提供了getConvertedData(data, type)方法来对item传入的数据做转换,方便拆包和提升item的复用性 支持viewpager的正常加载模式和懒加载 支持快速将listview适配器切换为recyclerView适配器 viewpager的notifyDataSetChanged可以正常更新界面 支持recyclerView的添加头部和底部 支持适配器的数据自动绑定,只用操作数据便可,adapter会自动notify界面零、重要接口adapter的item必须实现此接口,接口源码如下:public interface AdapterItem<T> {     /**      * @return item布局文件的layoutId      */     @LayoutRes     int getLayoutResId();     /**      * 初始化views      */     void bindViews(final View root);     /**      * 设置view的参数      */     void setViews();     /**      * 根据数据来设置item的内部views      *      * @param model    数据list内部的model      * @param position 当前adapter调用item的位置      */     void handleData(T model, int position); }例子:public class TextItem implements AdapterItem<DemoModel> {     @Override     public int getLayoutResId() {         return R.layout.demo_item_text;     }     TextView textView;     @Override     public void bindViews(View root) {         textView = (TextView) root.findViewById(R.id.textView);     }     @Override     public void setViews() { }     @Override     public void handleData(DemoModel model, int position) {         textView.setText(model.content);     } }一、ListView GridView通用适配器——CommonAdapter只需继承CommonAdapter便可实现适配器listView.setAdapter(new CommonAdapter<DemoModel>(data, 1) {     public AdapterItem<DemoModel> createItem(Object type) {         return new TextItem();     } });二、RecyclerView通用适配器——CommonRcvAdapter通过继承CommonRcvAdapter来实现适配器:mAdapter = new CommonRcvAdapter<DemoModel>(data) {  public AdapterItem createItem(Object type) {         return new TextItem();   } };三、ViewPager的通用适配器——CommonPagerAdapter通过继承CommonPagerAdapter来实现适配器viewPager.setAdapter(new CommonPagerAdapter<DemoModel>() {     public AdapterItem createItem(Object type) {         return new TextItem();     } });设计思路1. Adapter如果用adapter常规写法,你会发现代码量很大,可读性低。如果adapter中有多个类型的Item,我们还得在getView()中写很多if-else语句,很乱。 而现在我让adapter的代码量减少到一个8行的内部类,如果你需要更换item只需要动一行代码,真正实现了可插拔化。最关键的是item现在作为了一个独立的对象,可以方便的进行复用。2. AdapterItem和原来方式最为不同的一点就是我把adapter的item作为了一个实体,这种方式借鉴了RecyclerViewViewHolder的设计。把item作为实体的好处有很多,比如复用啊,封装啊,其余的就不细说了。3. 分层在使用过程中,我发现如果adapter放在view层,那就会影响到view层的独立性。此外adapter中经常有很多数据处理的操作,比如通过type选择item,数据的拆包、转换等操作。于是我还是推荐把adapter放在mvp的p层,或者是mvvm的m层。通过在实际的项目中使用来看,放在m或p层的效果较好,view的复用也比较好做。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值