文章来自Stormzhang

Android简便通用的SimpleBaseAdapter

序.在 android5.0 以后使用了 RecyledView ,后面出文章

在Android开发中经常用到ListView、GridView等列表,于是Adapter也就比较常用的了, 而实际项目中稍微复杂点的基本都会用到BaseAdapter, 大家都知道继承自BaseAdapter必须要重写getCount(), getItem(), getItemId(), getView()这几个方法, 而且可能为了优化列表的加载还还会经常采用ViewHoder模式, 试想列表多了估计都写烦了吧,那么今天就来教大家一种通用、简洁的Adapter,以后你就可以几行代码搞定一个复杂的Adapter了。

通用的SimpleBaseAdapter 首先我们来解决每次都重写BaseAdapter的那几个方法的问题,解决方案很简单,直接写一个抽象的SimpleBaseAdapter,代码如下

public abstract class SimpleBaseAdapter<T> extends BaseAdapter {

    protected Context context;
    protected List<T> data;

    public SimpleBaseAdapter(Context context, List<T> data) {
        this.context = context;
        this.data = data == null ? new ArrayList<T>() : new ArrayList<T>(data);
    }

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

    @Override
    public Object getItem(int position) {
        if (position >= data.size())
            return null;
        return data.get(position);
    }

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

上面代码我不就过多解释了,相信大家都能看懂。之后建立Adapter的时候只需要继承自SimpleBaseAdapter,把List数据源传进去,再重写getView方法就好了。

通用的ViewHolder 上面的SimpleBaseAdapter已经大大减少了不少的重复性工作,但是对于ViewHolder的优化依然还有很多重复性的代码要写,那么下面就来看看如何打造一个通用的ViewHolder。

关于ListView的优化这里再多说下吧,主要是两点:

  • 一是判断convertView为空的时候才会inflate,复用item view,

  • 二便是使用ViewHoder模式,通过convertView.setTag与convertView进行绑定,然后当convertView复用时,直接从与之对于的ViewHolder(getTag)中拿到convertView布局中的控件,省去了findViewById的时间~

下面看我们的ViewHolder类:

public class ViewHolder {
    private SparseArray<View> views = new SparseArray<View>();
    private View convertView;

    public ViewHolder(View convertView) {
        this.convertView = convertView;
    }

    public <T extends View> T getView(int resId) {
        View v = views.get(resId);
        if (null == v) {
            v = convertView.findViewById(resId);
            views.put(resId, v);
        }
        return (T) v;
    }
}

可以看到这里定义的ViewHolder代码很简单,使用起来也是非常的轻便。SparseArray在代码理解上等价于HashMap, SparseArray是Android提供的一个数据结构,但是比Map的效率更高。

最终的SimpleBaseAdapter 废话不多说了,我们结合以上两点,直接上个最终的SimpleBaseAdapter

public abstract class SimpleBaseAdapter<T> extends BaseAdapter {

    protected Context context;
    protected List<T> data;

    public SimpleBaseAdapter(Context context, List<T> data) {
        this.context = context;
        this.data = data == null ? new ArrayList<T>() : new ArrayList<T>(data);
    }

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

    @Override
    public Object getItem(int position) {
        if (position >= data.size())
            return null;
        return data.get(position);
    }

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

    /**
     * 该方法需要子类实现,需要返回item布局的resource id
     *
     * @return
     */
    public abstract int getItemResource();

    /**
     * 使用该getItemView方法替换原来的getView方法,需要子类实现
     *
     * @param position
     * @param convertView
     * @param parent
     * @param holder
     * @return
     */
    public abstract View getItemView(int position, View convertView, ViewHolder holder);

    @SuppressWarnings("unchecked")
    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;
        if (null == convertView) {
            convertView = View.inflate(context, getItemResource(), null);
            holder = new ViewHolder(convertView);
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }
        return getItemView(position, convertView, holder);
    }

    public class ViewHolder {
        private SparseArray<View> views = new SparseArray<View>();
        private View convertView;

        public ViewHolder(View convertView) {
            this.convertView = convertView;
        }

        @SuppressWarnings("unchecked")
        public <T extends View> T getView(int resId) {
            View v = views.get(resId);
            if (null == v) {
                v = convertView.findViewById(resId);
                views.put(resId, v);
            }
            return (T) v;
        }
    }

    public void addAll(List<T> elem) {
        data.addAll(elem);
        notifyDataSetChanged();
    }

    public void remove(T elem) {
        data.remove(elem);
        notifyDataSetChanged();
    }

    public void remove(int index) {
        data.remove(index);
        notifyDataSetChanged();
    }

    public void replaceAll(List<T> elem) {
        data.clear();
        data.addAll(elem);
        notifyDataSetChanged();
    }
}

使用 自己的Adapter需要继承SimpleBaseAdapter,并且实现getItemResource和getItemView两个方法就好了, 下面看下例子:

public class TestFoodListAdapter extends SimpleBaseAdapter<String> {

    public TestFoodListAdapter(Context context, List<String> data) {
        super(context, data);
    }

    @Override
    public int getItemResource() {
        return R.layout.listitem_test;
    }

    @Override
    public View getItemView(int position, View convertView, ViewHolder holder) {
        TextView text = holder.getView(R.id.text);
        text.setText(getItem(position));
        return convertView;
    }
}

相信大家都能看懂,具体的xml布局文件就不写出来了,上面的示例和google推荐的写法类似,只不过稍加变通之后减少了很多重复的代码, 是不是清爽了很多呢?