Android 通用ListView、GridView适配器

1、简述

在Android开发肯定避免不了与adapter打交道,一般都是继承于BaseAdapter重写里面几个方法,然后一个ListView对应一个Adapter,那自然项目中就出现一大堆Adapter,鉴于此Adapter出现大量的重复代码是否有办法可以简化呢?答案是肯定的。

本库地址:https://github.com/mylhyl/Android-CygAdapter

2、常规写法

只贴adapter部分,layout就不贴了,因为不是重点。

 

public class SimpleAdapter extends BaseAdapter {
    public static class ViewHolder {
        public TextView tvName;
    }

    private List<Student> studentList;
    private LayoutInflater inflater;

    public SimpleAdapter(Context context, List<Student> students) {
        inflater = LayoutInflater.from(context);
        this.studentList = students;
    }

    @Override
    public int getCount() {
        return studentList.size();
    }

    @Override
    public Student getItem(int position) {
        return studentList.get(position);
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;
        View view = convertView;
        if (view == null) {
            view = inflater.inflate(android.R.layout.simple_list_item_1, parent, false);
            holder = new ViewHolder();
            holder.tvName = (TextView) view.findViewById(android.R.id.text1);
            view.setTag(holder);
        } else {
            holder = (ViewHolder) view.getTag();
        }
        Student student = getItem(position);
        holder.tvName.setText(student.naem);
        return view;
    }
}

 

Activity中调用

 

setListAdapter(new SimpleAdapter(getContext(),datas));

 

这段代码应该不陌生,当继承BaseAdapter,必须重写getCount、getItem、getItemId、getView这4个方法,然后写一个构造方法传入实体对象。可以说你再写一个BaseAdapter也是同样的写法,仔细的你应该会发现,只有构造方法实体对象与getView中方法体业务逻辑有差异嘛,那么是不是可以把实体对象改写成泛型,getView方法体业务逻辑抽象出来,在外实现呢?那是必须可以的,接下来我们一步一步改善这要命的重复代码。

3、实体对象改成泛型

新建一个adapter命名为CygAdapter,有变化的代码加了注释。

public class CygAdapter<T> extends BaseAdapter {//变化
    public static class ViewHolder {
        public TextView tvName;
    }

    private List<T> studentList;//变化
    private LayoutInflater inflater;

    public CygAdapter(Context context, List<T> students) {
        inflater = LayoutInflater.from(context);
        this.studentList = students;
    }

    @Override
    public int getCount() {
        return studentList.size();
    }

    @Override
    public T getItem(int position) {//变化
        return studentList.get(position);
    }

    @Override
    public long getItemId(int position) {
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;
        View view = convertView;
        if (view == null) {
            view = inflater.inflate(android.R.layout.simple_list_item_1, parent, false);
            holder = new ViewHolder();
            holder.tvName = (TextView) view.findViewById(android.R.id.text1);
            view.setTag(holder);
        } else {
            holder = (ViewHolder) view.getTag();
        }
        T student = getItem(position);//变化
        holder.tvName.setText(student.naem);//这里出问题咯,这个T泛型类根本就没有name变量嘛,怎么办?
        return view;
    }
}

上面代码已成功把实体对象改成了泛型,但是getView中的业务逻辑这时出现了上面代码中的问题,无法取到name变量,难道给Student实体类加注解?对于一个Student类注解是行得通滴,加一个@name注解嘛,但是又来一个学校School类、班级Class类那岂不是加N个不同命名的注解?这样一来加注解方案肯定就走不下去了。那么我们可以尝试把getView的业务逻辑抽象出来在外部实现,那现在就需要一个抽象方法了。

4、抽象getView的业务逻辑

我们再CygAdapter改造一下,新建一个抽象方法 onBindData(ViewHolder viewHolder, final T item, final int position);

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;
        View view = convertView;
        if (view == null) {
            view = inflater.inflate(android.R.layout.simple_list_item_1, parent, false);
            holder = new ViewHolder();
            holder.tvName = (TextView) view.findViewById(android.R.id.text1);
            view.setTag(holder);
        } else {
            holder = (ViewHolder) view.getTag();
        }
        T student = getItem(position);
        //holder.tvName.setText(student.naem);//这里出问题咯,怎么办?
        onBindData(holder,student,position);//把上面holder.tvName.setText(student.naem)抽象到外部实现
        return view;
    }
    //新建抽象方法
    public abstract void onBindData(ViewHolder viewHolder, final T item, final int position);

再来看在Activity中用法

        setListAdapter(new com.mylhyl.cygadapter.sample.blog.CygAdapter<Student>(getContext(),datas) {
            @Override
            public void onBindData(ViewHolder viewHolder, Student item, int position) {
                viewHolder.tvName.setText(item.naem);
            }
        });

换一个School对象试试

        ArrayList<School> datas = new ArrayList();
        datas.add(new School("珠海北师大"));
        datas.add(new School("长沙湖南大学"));
        setListAdapter(new com.mylhyl.cygadapter.sample.blog.CygAdapter<School>(getContext(), datas) {
            @Override
            public void onBindData(ViewHolder viewHolder, School item, int position) {
                viewHolder.tvName.setText(item.address);
            }
        });

是不是爽YY了,不管换什么实体对象照样搞定你。Student与School二个实体类都分别只有一个属性name、address,那么现在需求上变化,在Student类中新增一个age年龄属性,显示在ListView上。这样一来我们的layout就得新加一个TextView来用显示age,我们就不新建了利用系统自带的android.R.layout.simple_list_item_2,于是就可以开干了,重新构造List<Student>数据集啥的

 

        ArrayList<Student> datas = new ArrayList();
        datas.add(new Student(21,"张三"));
        datas.add(new Student(22,"李四"));
        setListAdapter(new com.mylhyl.cygadapter.sample.blog.CygAdapter<Student>(getContext(), datas) {
            @Override
            public void onBindData(ViewHolder viewHolder, Student item, int position) {
                viewHolder.tvName.setText(item.naem);
                viewHolder.tvAge.setText(String.valueOf(item.age));//尼玛没有ViewHolder中没有tvAge呀
            }
        });

上面代码ViewHolder中没有tvAge呀,新建一个不就得了,这样肯定是不行的,与刚才说的加注解方案一样的道理,ViewHolder中的变量只会越来越多,怎么办呢?竟然问题出现在ViewHolder就得从他来下手,我们看getView中的那段ViewHolder相关的代码

 

没有加age之前

        ViewHolder holder;
        View view = convertView;
        if (view == null) {
            view = inflater.inflate(android.R.layout.simple_list_item_1, parent, false);//一个TextView
            holder = new ViewHolder();
            holder.tvName = (TextView) view.findViewById(android.R.id.text1);//名字
            view.setTag(holder);
        } else {
            holder = (ViewHolder) view.getTag();
        }

加age之后

        ViewHolder holder;
        View view = convertView;
        if (view == null) {
            view = inflater.inflate(android.R.layout.simple_list_item_2, parent, false);//二个TextView
            holder = new ViewHolder();
            holder.tvName = (TextView) view.findViewById(android.R.id.text1);//名字
            holder.tvAge = (TextView) view.findViewById(android.R.id.text2);//年龄
            view.setTag(holder);
        } else {
            holder = (ViewHolder) view.getTag();
        }

上面代码是不是看出点什么呢?是不是除了inflate与findViewById部分不一样,那是不是有点思路了?新建一个类抽出相同部分,写成一个通用的ViewHolder,开干吧!

5、通用ViewHolder

这个是参考了鸿洋的

新建一个不可继承的CygViewHolder,首先看getView中代码,这个CygViewHolder类肯定不是每都new,只有当convertView为null才new的,所以先搞个静态方法用于取得CygViewHolder,为了不被外部直接new CygViewHolder,我们得把构造方法变为私用。来啊来呀,看改造后的代码啊!

CygViewHolder

public final class CygViewHolder {
    private Context mContext;
    private View mItemView;

    public static CygViewHolder get(Context context, int resource, View convertView, ViewGroup parent) {
        if (convertView == null) {
            View view = LayoutInflater.from(context).inflate(resource, parent, false);
            return new CygViewHolder(context, view);
        }
        return (CygViewHolder) convertView.getTag();
    }

    private CygViewHolder(Context context, View itemView) {
        mContext = context;
        mItemView = itemView;
        mItemView.setTag(this);
    }

    public View getItemView() {
        return mItemView;
    }
}

CygAdapter

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        //没有了一堆重复代码
        CygViewHolder viewHolder = CygViewHolder.get(context, resource, convertView, parent);
        T student = getItem(position);
        onBindData(viewHolder, student, position);
        return viewHolder.getItemView();
    }

    //ViewHolder改为CygViewHolder
    public abstract void onBindData(CygViewHolder viewHolder, final T item, final int position);

ListView绑定CygAdapter的时候出现问题了,无法查找控件

        ArrayList<Student> datas = new ArrayList();
        datas.add(new Student(21,"张三"));
        datas.add(new Student(22,"李四"));
        setListAdapter(new com.mylhyl.cygadapter.sample.blog.CygAdapter<Student>(getContext(), datas) {
            @Override
            public void onBindData(CygViewHolder viewHolder, Student item, int position) {
                //查找TextView,还是没有tvName\tvAge?又得如何处理,想个办法
                viewHolder.tvName = (TextView) view.findViewById(android.R.id.text1);
                viewHolder.tvAge = (TextView) view.findViewById(android.R.id.text2);
                //为TextView设置数据
                viewHolder.tvName.setText(item.naem);
                viewHolder.tvAge.setText(String.valueOf(item.age));
            }
        });

上面代码onBindData绑定数据代码段的问题如何处理呢?思路是这样,既然CygViewHolder中已经有了每个Item的View,那么我们可以在CygViewHolder写个findViewById方法查找控件,并缓存在SparseArray<View>中,如果有直接从缓存中返回,否则通过itemView.findViewById查找,缓存再返回。

public final class CygViewHolder {
    private Context mContext;
    private SparseArray<View> mViews;//变化
    private View mItemView;

    public static CygViewHolder get(Context context, int resource, View convertView,
                                    ViewGroup parent) {
        if (convertView == null) {
            View view = LayoutInflater.from(context).inflate(resource, parent, false);
            return new CygViewHolder(context, view);
        }
        return (CygViewHolder) convertView.getTag();
    }

    private CygViewHolder(Context context, View itemView) {
        mContext = context;
        mViews = new SparseArray<>();//变化
        mItemView = itemView;
        mItemView.setTag(this);
    }


    public <T extends View> T findViewById(int id) {//新增方法
        View view = mViews.get(id);
        if (view == null) {
            view = mItemView.findViewById(id);
            mViews.put(id, view);
        }
        return (T) view;
    }
}

使用

        ArrayList<Student> datas = new ArrayList();
        datas.add(new Student(21, "张三"));
        datas.add(new Student(22, "李四"));
        setListAdapter(new com.mylhyl.cygadapter.sample.blog.CygAdapter<Student>
                (getContext(), android.R.layout.simple_list_item_2, datas) {
            @Override
            public void onBindData(CygViewHolder viewHolder, Student item, int position) {
                TextView tvName = viewHolder.findViewById(android.R.id.text1);
                TextView tvAge = viewHolder.findViewById(android.R.id.text2);
                tvName.setText(item.naem);
                tvAge.setText(String.valueOf(item.age));
            }
        });


 

现在是不是简单多了,adapter通用,只需使用时在onBindData中查到控件、给控件绑定数据,就这么简单。

内容比较多写的啰嗦了点,但出发点是好的,请耐心阅读,勿喷哦!!!

本库地址:https://github.com/mylhyl/Android-CygAdapter

说明一下CygViewHolder中我只加入了TextView.setText的方法,如有需要增加其它方法可以加Q群:435173211 

也可到GitHub先fork后再new pull request 可者Issues

 

欢迎关注公众号,每天为您推送“人生感悟、说话技巧、职场规则、职场成长”

 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

易家兴

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值