Android快速开发之设计通用的ListView,GridView的适配器

概述

在Android开发的时候,经常会使用ListView或GridView来显示数据,这时就需要写对应的适配器Adapter。如果针对每个ListView或者GridView都去创建一个适配器时,那么就会出现一大堆的重复的代码。为了体现代码的简洁性,我们可以设计一个通用的适配器来避免里面重复的逻辑。

常规实现

我们先列出常规适配器的使用方法,通过分析常见的使用方式,来分析设计出通用的适配器。
1.布局文件first_activity.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="horizontal" >

    <ListView
        android:id="@+id/listView"
        android:layout_width="match_parent"
        android:layout_height="match_parent" >
    </ListView>

</LinearLayout>

2.Activity类中代码

package com.yuminfeng.test;

import java.util.ArrayList;
import java.util.List;
import com.yuminfeng.adapter.MyAdapter;
import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;

public class FirstActivity extends Activity{

    private ListView listView;
    private List<String> datas; //listview中的数据源
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.first_activity);

        listView = (ListView) findViewById(R.id.listView);
        datas = new ArrayList<String>();
        for (int i = 0; i < 5; i++) {
            datas.add("Hello "+i);
        }
        MyAdapter adapter = new MyAdapter(this);
        adapter.setDataSource(datas);
        listView.setAdapter(adapter);
    }

}

3.ListView中item的布局文件item_listview.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" >

    <TextView
        android:id="@+id/textView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:textSize="20sp" />

</LinearLayout>

4.ListView的自定义Adapter类MyAdapter.java

package com.yuminfeng.adapter;

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

import com.yuminfeng.test.R;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;

public class MyAdapter extends BaseAdapter{

    private Context context;

    private List<String> datas = new ArrayList<String>();

    public MyAdapter(Context context){
        this.context = context;
    }

    /**
     * 添加数据源
     * @param datas
     */
    public void setDataSource(List<String> datas) {
        setDataSource(datas,true);
    }

    /**
     * 添加数据源
     * @param datas
     * @param isClear true:清除之前的数据源,false:追加数据源
     */
    public void setDataSource(List<String> datas,boolean isClear) {
        if(isClear){
            this.datas.clear();
        }
        this.datas.addAll(datas);
        notifyDataSetChanged();
    }

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

    @Override
    public Object getItem(int position) {
        return datas.get(position);
    }

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

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder viewHolder = null;
        if(convertView == null){
            convertView = LayoutInflater.from(context).inflate(R.layout.item_listview, parent,false);
            viewHolder = new ViewHolder(convertView);
            convertView.setTag(viewHolder);
        }else{
            viewHolder = (ViewHolder) convertView.getTag();
        }
        viewHolder.textView.setText(datas.get(position));
        return convertView;
    }

    public class ViewHolder{
        private TextView textView;
        public ViewHolder(View convertView){
            this.textView = (TextView) convertView.findViewById(R.id.textView);
        }
    }
}

以上就是实现ListView显示数据的基本写法,如果有10个ListView那么就会有10个这样的适配器,这10个适配器中存在大量重复的代码。为了避免这种情况,我们需要设计出一个通用的适配器,在这个通用的适配器的基础上,我们在进行不同的业务操作。

设计通用的适配器

首先分析Adapter中的包含元素:
填充布局时,需要用到Context;
需要List来存储显示的数据源;
需要布局文件来作为ListView的item;
需要ViewHolder类来优化ListView的性能;
以上几项,都是根据不同的ListView,来做不同的处理,其他重写的BaseAdapter的方法都是固定的。所以只要抽出上述的参数,灵活的进行处理,这就是设计通用的适配器的思想。
了解完了之后,我们的代码如下:

package com.yuminfeng.adapter;

import java.util.ArrayList;
import java.util.List;
import android.content.Context;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;

public abstract class CommonAdapter<E> extends BaseAdapter {

    //context
    private Context context;
    //数据源
    private List<E> mDatas = new ArrayList<E>();
    //item的layout布局 id
    private int layoutResId;

    public CommonAdapter(Context context,int layoutResId) {
        this.context = context;
        this.layoutResId = layoutResId;
    }

    /**
     * 添加数据源
     * @param data
     */
    public void setDataSource(List<E> data) {
        setDataSource(data, true);
    }

    /**
     * 添加数据源
     * @param data
     * @param isClear true:清除之前的数据源,false:追加数据源
     */
    public void setDataSource(List<E> data, boolean isClear) {
        if(isClear){
            mDatas.clear();
        }
        mDatas.addAll(data);
        notifyDataSetChanged();
    }

    /**
     * 只添加一个数据
     * @param data
     */
    public void addData(E data){
        mDatas.add(data);
        notifyDataSetChanged();
    }

    /**
     * 移除对象data的数据
     * @param data
     */
    public void removeData(E data){
        mDatas.remove(data);
        notifyDataSetChanged();
    }

    /**
     * 通过对象移除一条数据
     * @param position
     */
    public void removeData(int position){
        mDatas.remove(position);
        notifyDataSetChanged();
    }

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

    @Override
    public E getItem(int position) {
        return mDatas.get(position);
    }

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

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        BaseViewHolder viewHolder = null;
        if(convertView == null){
            viewHolder = createViewHolder(parent,layoutResId);
            if(viewHolder == null || viewHolder.getConvertView() == null){
                throw new NullPointerException("createViewHolder不能返回null");
            }
            convertView = viewHolder.getConvertView();
            convertView.setTag(viewHolder);
        }else{
            viewHolder = (BaseViewHolder) convertView.getTag();
        }
        bindViewHolder(viewHolder, position, getItem(position));

        return viewHolder.getConvertView();
    }

    private BaseViewHolder createViewHolder(ViewGroup parent,int layoutResId){
        View convertView = LayoutInflater.from(context).inflate(layoutResId, parent,false);
        return new BaseViewHolder(convertView);
    }

    protected abstract void bindViewHolder(BaseViewHolder viewHolder,int position,E data);

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

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

        public View getConvertView(){
            return convertView;
        }

        /**
         * 根据View的id,取到View的对象
         * @param viewId
         * @return
         */
        @SuppressWarnings("unchecked")
        public <V> V getView(int viewId){
            View view = views.get(viewId);
            if(view == null){
                view = convertView.findViewById(viewId);
                views.put(viewId, view);
            }
            return (V) view;
        }
    }

}

上面便是设计的一个通用的适配器类,这是个抽象类。
实现时,首先继承这个类,然后重写里面的抽象方法bindViewHolder。这个方法是开放给外部的回调方法,用来设置具体的item数据。还有在创建这个子类的对象时,传入两个参数:上下文context,和布局item的id。
我们来看它的子类对象,如下:

package com.yuminfeng.adapter;

import java.util.Map;
import com.yuminfeng.test.R;
import android.content.Context;
import android.widget.ImageView;
import android.widget.TextView;

public class MyAdapter extends CommonAdapter<Map<String, Object>>{

    public MyAdapter(Context context, int layoutResId) {
        super(context, layoutResId);
    }

    @Override
    protected void bindViewHolder(BaseViewHolder viewHolder,int position, Map<String, Object> data) {
        TextView title = viewHolder.getView(R.id.title);
        TextView content = viewHolder.getView(R.id.content);
        TextView time = viewHolder.getView(R.id.time);
        ImageView image = viewHolder.getView(R.id.imageView);

        title.setText((String)data.get("title"));
        content.setText((String)data.get("content"));
        time.setText((String)data.get("time"));
        image.setImageResource((Integer)data.get("icon"));
    }

}

上面的类非常简单,就是给具体的item中的View设置值,其它继承CommonAdapter类的子类,也是一样的操作。
我们再看Activity中的调用,如下:

package com.yuminfeng.test;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import com.yuminfeng.adapter.MyAdapter;
import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;

public class FirstActivity extends Activity{

    private ListView listView;
    private List<Map<String, Object>> datas; //listview中的数据源
    private int[] imageIds = new int[]{R.drawable.pic1,R.drawable.pic2,R.drawable.pic3,R.drawable.pic4,R.drawable.pic5};
     @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.first_activity);

        listView = (ListView) findViewById(R.id.listView);
        datas = new ArrayList<Map<String, Object>>();

        for (int i = 0; i < imageIds.length; i++) {
            Map<String, Object> map = new HashMap<String, Object>();
            map.put("icon",imageIds[i] );
            map.put("title","美女"+i+"只");
            map.put("content","这是内容,这是内容");
            map.put("time","20160428");
            datas.add(map);
        }
        MyAdapter adapter = new MyAdapter(this, R.layout.item_listview_more);
        adapter.setDataSource(datas);

        listView.setAdapter(adapter);
    }
}

以上也是非常简单的逻辑,创建MyAdapter类,传入context和布局item。然后设置数据源后,直接使用即可。
这样就将之前重复的逻辑全部封装到抽象类CommonAdapter中,现在不需要关心了,继承该类后,只需绑定item中每个View的数据就可以了。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值