打造萬能Adapter(適配器)——適用於ListView、GridListView

 

一、利用普通的Adapter實現ListView列表——這是最基礎的適配器

以下代碼是最普通的實現方法:

1、MainActiviy.java

 

public class MainActivity extends Activity {

    private List mData = new ArrayList(Arrays.asList("HongYang", "GuoLin", "RenYuGang", "Jiatao"));
    private Context context;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        context = this;
        setContentView(R.layout.activity_main);
        ListView listView = (ListView) findViewById(R.id.listview);

        MyAdapter mAdapter = new MyAdapter(context, mData);
        listView.setAdapter(mAdapter);
    }
}

 

2、activity_main.xml

 




    

 

3、MyAdapter.java

 

/**
 * Created by jiatao on 2016/6/10.
 * 這是最普通的適配器:
 *      MyAdapter繼承BaseAdapter,然後getView裡面使用ViewHolder模式;
 *      一般情況下,我們的寫法是這樣的:對於不同佈局的ListView,我們會有一個對應的Adapter,在Adapter中又會有一個ViewHolder類來提高效率。
 *      這樣出現ListView就會出現與之對應的Adapter類、ViewHolder類;
 *  那麼有沒有辦法減少我們的編碼呢?第二部分中將初步實現
 */
public class MyAdapter extends BaseAdapter {

    private LayoutInflater mLayoutInflater;
    private Context mContext;
    private List mData;

    public MyAdapter(Context mContext, List mData) {
        mLayoutInflater = LayoutInflater.from(mContext);
        this.mContext = mContext;
        this.mData = mData;
    }

    @Override
    public int getCount() {
        return mData == null ? 0 : mData.size();
    }

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

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

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;
        if (convertView == null) {
            holder = new ViewHolder();
            convertView = mLayoutInflater.inflate(R.layout.item_string_listview, parent, false);
            holder.tv_title = (TextView) convertView.findViewById(R.id.tv_title);
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }

        holder.tv_title.setText(mData.get(position));
        return convertView;
    }

    class ViewHolder {
        private TextView tv_title;
    }
}

 

4、item_string_listview.xml

 




 

二、打造通用的ViewHolder——使能在外部獲得ViewHolder、相關View控件、convertView

 

1、MainActiviy.java

同上

2、activity_main.xml

同上

3、ViewHolder.java

 

/**
 * Created by jiatao on 2016/6/11.
 * 通用的ViewHolder
 * 1、ViewHolder的作用是通過convertView.setTag與convertView進行綁定,然後當convertView復用時,直接從與之對於的ViewHolder(getTag)中拿到convertView佈局中的控件,省去瞭findViewById的時間;
 * 2、也就是說,實際上們每個convertView會綁定一個ViewHolder對象,這個viewHolder主要用於幫convertView存儲佈局中的控件;
 * 3、那麼我們隻要寫出一個通用的ViewHolder,然後對於任意的convertView,提供一個對象讓其setTag即可;
 * 4、既然是通用,那麼我們這個ViewHolder就不可能含有各種控件的成員變量瞭,因為每個Item的佈局是不同的,最好的方式是什麼呢?
 * 5、提供一個容器,專門存每個Item佈局中的所有控件,而且還要能夠查找出來;既然需要查找,那麼ListView肯定是不行瞭,需要一個鍵值對進行保存,鍵為控件的Id,值為控件的引用;
 * 6、相信大傢立刻就能想到Map;但是我們不用Map,因為有更好的替代類,就是我們android提供的SparseArray這個類,和Map類似,但是比Map效率,不過鍵隻能為Integer.
 */
public class ViewHolder {

    private SparseArray mViews;
    private View mConvertView;

    //構造函數
    private ViewHolder(Context context, int resLayoutId, ViewGroup parent) {
        this.mViews = new SparseArray();
        this.mConvertView = LayoutInflater.from(context).inflate(resLayoutId, parent, false);
        this.mConvertView.setTag(this);
    }

    //獲取一個ViewHolder
    public static ViewHolder getHolder(Context context, int resLayoutId, View convertView, ViewGroup parent) {
        if (convertView == null) {
            return new ViewHolder(context, resLayoutId, parent);
        }
        return (ViewHolder) convertView.getTag();
    }

    //通過控件的id獲取對應的控件,如果沒有則加入mViews;記住  T 這種用法
    public  T getItemView(int viewId) {
        View view = mViews.get(viewId);
        if (view == null) {
            view = mConvertView.findViewById(viewId);
            mViews.put(viewId, view);
        }
        return (T) view;
    }

    //獲得一個convertView
    public View getmConvertView() {
        return mConvertView;
    }

}

 

4、MyAdapter.java

 

/**
 * Created by jiatao on 2016/6/10.
 * 這是使用瞭通用的ViewHolder的適配器
 *  除瞭getView,其他方法都一樣:
 *      1、調用ViewHolder的get方法:
 *          如果convertView為null,new一個ViewHolder實例,通過使用mInflater.inflate加載佈局,然後new一個SparseArray用於存儲View,最後setTag(this);
 *          如果存在那麼直接getTag;
 *      2、通過getView(id)獲取控件,如果存在則直接返回,否則調用findViewById,返回存儲,返回。
 *      3、最後通過holder.getmConvertView()返回視圖;
 */
public class MyAdapter extends BaseAdapter {

    private LayoutInflater mLayoutInflater;
    private Context mContext;
    private List mData;

    public MyAdapter(Context mContext, List mData) {
        mLayoutInflater = LayoutInflater.from(mContext);
        this.mContext = mContext;
        this.mData = mData;
    }

    @Override
    public int getCount() {
        return mData == null ? 0 : mData.size();
    }

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

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

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        //實例化一個ViewHolder
        ViewHolder holder = ViewHolder.getHolder(mContext, R.layout.item_string_listview, convertView, parent);
        //獲取控件
        TextView view = holder.getItemView(R.id.tv_title);
        //給控件賦值
        view.setText(mData.get(position));
        return holder.getmConvertView();
    }
}

 

5、item_string_listview.xml

同上

 

三、初步打造通用的Adapter——通過泛型支持所有數據集合,抽取公共方法封裝成抽象基類,對外開發getView方法

 

1、MainActiviy.java

同上

2、activity_main.xml

同上

3、ViewHolder.java

同上

4、ComAdapter.java

 

/**
 * Created by jiatao on 2016/6/11.
 * 通用的適配器
 * 1、ComAdapter需要保持一個List對象,存儲一個Bean的集合,不同的ListView,Bean肯定是不同的,這個ComAdapter肯定需要支持泛型,內部維持一個List
 * 2、ComAdapter是一個抽象類,提取getView以外的方法,getView方法留給子類去實現
 */
public abstract class ComAdapter extends BaseAdapter {

    protected LayoutInflater mLayoutInflater;
    protected Context mContext;
    protected List mData;

    public ComAdapter(Context mContext, List mData) {
        mLayoutInflater = LayoutInflater.from(mContext);
        this.mContext = mContext;
        this.mData = mData;
    }

    @Override
    public int getCount() {
        return mData == null ? 0 : mData.size();
    }

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

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

5、MyAdapter.java

 

 

/**
 * Created by jiatao on 2016/6/11.
 * 繼承ComAdapter抽象類,而且使用瞭通用的ViewHolder的適配器
 */
public class MyAdapter extends ComAdapter {

    public MyAdapter(Context mContext, List mData){
        super(mContext, mData);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        //實例化一個ViewHolder
        ViewHolder holder = ViewHolder.getHolder(mContext, R.layout.item_string_listview, convertView, parent);
        //獲取控件
        TextView view = holder.getItemView(R.id.tv_title);
        //給控件賦值
        view.setText((String) mData.get(position));
        return holder.getmConvertView();
    }
}

6、item_string_listview.xml

 

同上

四、進一步打造通用的Adapter——使用匿名內部類獲得適配器

 

1、MainActiviy.java

 

public class MainActivity extends Activity {

    private List mData = new ArrayList(Arrays.asList("HongYang", "GuoLin", "RenYuGang", "Jiatao"));
    private Context context;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        context = this;
        setContentView(R.layout.activity_main);
        ListView listView = (ListView) findViewById(R.id.listview);

        //直接使用匿名內部類即可取得適配器
        ComAdapter mAdapter = new ComAdapter(context, mData, R.layout.item_string_listview) {
            @Override
            public void convert(ViewHolder holder, String item) {//通過ViewHolder找到iew,通過Item設置值
                TextView view =holder.getItemView(R.id.tv_title);
                view.setText(item);
            }
        };
        listView.setAdapter(mAdapter);
    }
}

2、activity_main.xml

 

同上

 

3、ViewHolder.java

同上

4、ComAdapter.java

/**
 * Created by jiatao on 2016/6/11.
 * 通用的適配器
 * 1、所有的Adapter的第一行(ViewHolder viewHolder = getViewHolder(position, convertView,parent);)和 最後一行:return viewHolder.getConvertView();一定是一樣的。
 * 2、那麼我們可以這樣做:我們把第一行和最後一行寫死,把中間變化的部分抽取出來,這不就是OO的設計原則嘛。
 */
public abstract class ComAdapter extends BaseAdapter {

    protected LayoutInflater mLayoutInflater;
    protected Context mContext;
    protected List mData;
    protected int mItemLayoutId;

    public ComAdapter(Context mContext, List mData, int mItemLayoutId) {
        mLayoutInflater = LayoutInflater.from(mContext);
        this.mContext = mContext;
        this.mData = mData;
        this.mItemLayoutId = mItemLayoutId;
    }

    @Override
    public int getCount() {
        return mData == null ? 0 : mData.size();
    }

    @Override
    public T getItem(int position) {
        return mData.get(position);
    }

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

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        //實例化一個ViewHolder
        ViewHolder holder = getViewHolder(position, convertView, parent);
        //使用對外公開的convert方法,通過ViewHolder把View找到,通過Item設置值
        convert(holder, getItem(position));
        return holder.getmConvertView();
    }

    private ViewHolder getViewHolder(int position, View convertView, ViewGroup parent){
        return ViewHolder.getHolder(mContext, mItemLayoutId, convertView, parent);
    }

    /**
     * 對外公佈瞭一個convert方法,並且還把ViewHolder和本item對應的Bean對象給傳出去
     * 現在convert方法裡面需要幹嘛呢?通過ViewHolder把View找到,通過Item設置值
     */
    public abstract void convert(ViewHolder holder, T item);

}

5、MyAdapter.java

不再使用,刪除即可

6、item_string_listview.xml

同上

五、打造終極版通用Adapter——進一步完善ViewHolder使匿名內部類中convert方法隻需一行代碼即能給對應的view賦值

 

1、MainActiviy.java

 

/**
 * 通過匿名內部類獲得適配器,適配內的convert方法隻需一行代碼即可為對應的view賦值
 */
public class MainActivity extends Activity {

    private List mData = new ArrayList(Arrays.asList("HongYang", "GuoLin", "RenYuGang", "Jiatao"));
    private Context context;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        context = this;
        setContentView(R.layout.activity_main);
        ListView listView = (ListView) findViewById(R.id.listview);

        //直接使用匿名內部類即可取得適配器
        ComAdapter mAdapter = new ComAdapter(context, mData, R.layout.item_string_listview) {
            @Override
            public void convert(ViewHolder holder, String item) {//通過ViewHolder找到view,通過Item設置值
                holder.setTextView(R.id.tv_title, item);//隻需一行代碼即可實現:給對應的view賦值
            }
        };
        listView.setAdapter(mAdapter);
    }
}

2、activity_main.xml

 

同上

 

3、ViewHolder.java 

  /**

  * Created by jiatao on 2016/6/11.

  * 通用的ViewHolder

  * 1、ViewHolder的作用是通過convertView.setTag與convertView進行綁定,然後當convertView復用時,直接從與之對於的ViewHolder(getTag)中拿到convertView佈局中的控件,省去瞭findViewById的時間;

  * 2、也就是說,實際上們每個convertView會綁定一個ViewHolder對象,這個viewHolder主要用於幫convertView存儲佈局中的控件;

  * 3、那麼我們隻要寫出一個通用的ViewHolder,然後對於任意的convertView,提供一個對象讓其setTag即可;

  * 4、既然是通用,那麼我們這個ViewHolder就不可能含有各種控件的成員變量瞭,因為每個Item的佈局是不同的,最好的方式是什麼呢?

  * 5、提供一個容器,專門存每個Item佈局中的所有控件,而且還要能夠查找出來;既然需要查找,那麼ListView肯定是不行瞭,需要一個鍵值對進行保存,鍵為控件的Id,值為控件的引用;

  * 6、相信大傢立刻就能想到Map;但是我們不用Map,因為有更好的替代類,就是我們android提供的SparseArray這個類,和Map類似,但是比Map效率,不過鍵隻能為Integer.

  * 


  * 最後的封裝:

  * 我們現在在convertView裡面需要這樣:

  *

  * @Override public void convert(ViewHolder viewHolder, String item) {

  * TextView view = viewHolder.getView(R.id.id_tv_title);

  * view.setText(item);

  * }

  * 我們細想一下,其實佈局裡面的View常用也就那麼幾種:ImageView,TextView,Button,CheckBox等等;

  * 那麼我覺得ViewHolder還可以封裝一些常用的方法,比如setText(id,String);setImageResource(viewId, resId);setImageBitmap(viewId, bitmap);

  */

  public class ViewHolder {

  private SparseArray mViews;

  private View mConvertView;

  //構造函數

  private ViewHolder(Context context, int resLayoutId, ViewGroup parent) {

  this.mViews = new SparseArray();

  this.mConvertView = LayoutInflater.from(context).inflate(resLayoutId, parent, false);

  this.mConvertView.setTag(this);

  }

  //獲取一個ViewHolder

  public static ViewHolder getHolder(Context context, int resLayoutId, View convertView, ViewGroup parent) {

  if (convertView == null) {

  return new ViewHolder(context, resLayoutId, parent);

  }

  return (ViewHolder) convertView.getTag();

  }

  //通過控件的id獲取對應的控件,如果沒有則加入mViews;記住 T 這種用法

  public T getItemView(int viewId) {

  View view = mViews.get(viewId);

  if (view == null) {

  view = mConvertView.findViewById(viewId);

  mViews.put(viewId, view);

  }

  return (T) view;

  }

  //獲得一個convertView

  public View getmConvertView() {

  return mConvertView;

  }

  /**

  * 為TextView賦值

  */

  public ViewHolder setTextView(int viewId, String text) {

  TextView view = getItemView(viewId);

  view.setText(text);

  return this;

  }

  /**

  * 為ImageView賦值——drawableId

  */

  public ViewHolder setImageResource(int viewId, int drawableId) {

  ImageView view = getItemView(viewId);

  view.setImageResource(drawableId);

  return this;

  }

  /**

  * 為ImageView賦值——bitmap

  */

  public ViewHolder setImageBitmap(int viewId, Bitmap bitmap) {

  ImageView view = getItemView(viewId);

  view.setImageBitmap(bitmap);

  return this;

  }

  }

4、ComAdapter.java

同上

5、MyAdapter.java

不再使用,刪除即可

6、item_string_listview.xml

同上

 

六、實例運用

1、MainActiviy.java

 

/**
 * 運用萬能適配器實現復雜視圖的實例
 */
public class MainActivity extends Activity {

    private List beanlist;
    private Context context;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        context = this;
        setContentView(R.layout.activity_main);
        ListView listView = (ListView) findViewById(R.id.listview);

        beanlist = new ArrayList();
        for(int i=1;i<10;i++){
            ArticleBean bean = new ArticleBean("文章"+i,"內容"+i+":周三早上丟失瞭紅色錢包,在食堂二樓","2016060"+i,"1718013691"+i);
            beanlist.add(bean);
        }

        //直接使用匿名內部類即可取得適配器
        ComAdapter mAdapter = new ComAdapter(context, beanlist, R.layout.item_example_listview) {
            @Override
            public void convert(ViewHolder holder, ArticleBean item) {//通過ViewHolder找到view,通過Item設置值
                holder.setTextView(R.id.tv_title, item.getTitle());
                holder.setTextView(R.id.tv_describe, item.getContent());
                holder.setTextView(R.id.tv_time, item.getTime());
                holder.setTextView(R.id.tv_phone, item.getPhone());
            }
        };
        listView.setAdapter(mAdapter);
    }
}

2、activity_main.xml

 

 




    

3、ViewHolder.java

 

 

 

public class ViewHolder {

    private SparseArray mViews;
    private View mConvertView;

    //構造函數
    private ViewHolder(Context context, int resLayoutId, ViewGroup parent) {
        this.mViews = new SparseArray();
        this.mConvertView = LayoutInflater.from(context).inflate(resLayoutId, parent, false);
        this.mConvertView.setTag(this);
    }

    //獲取一個ViewHolder
    public static ViewHolder getHolder(Context context, int resLayoutId, View convertView, ViewGroup parent) {
        if (convertView == null) {
            return new ViewHolder(context, resLayoutId, parent);
        }
        return (ViewHolder) convertView.getTag();
    }

    //通過控件的id獲取對應的控件,如果沒有則加入mViews;記住  T 這種用法
    public  T getItemView(int viewId) {
        View view = mViews.get(viewId);
        if (view == null) {
            view = mConvertView.findViewById(viewId);
            mViews.put(viewId, view);
        }
        return (T) view;
    }

    //獲得一個convertView
    public View getmConvertView() {
        return mConvertView;
    }

    /**
     * 為TextView賦值
     */
    public ViewHolder setTextView(int viewId, String text) {
        TextView view = getItemView(viewId);
        view.setText(text);
        return this;
    }

    /**
     * 為ImageView賦值——drawableId
     */
    public ViewHolder setImageResource(int viewId, int drawableId) {
        ImageView view = getItemView(viewId);
        view.setImageResource(drawableId);
        return this;
    }

    /**
     * 為ImageView賦值——bitmap
     */
    public ViewHolder setImageBitmap(int viewId, Bitmap bitmap) {
        ImageView view = getItemView(viewId);
        view.setImageBitmap(bitmap);
        return this;
    }

}

4、ComAdapter.java

 

public abstract class ComAdapter extends BaseAdapter {

    protected LayoutInflater mLayoutInflater;
    protected Context mContext;
    protected List mData;
    protected int mItemLayoutId;

    public ComAdapter(Context mContext, List mData, int mItemLayoutId) {
        mLayoutInflater = LayoutInflater.from(mContext);
        this.mContext = mContext;
        this.mData = mData;
        this.mItemLayoutId = mItemLayoutId;
    }

    @Override
    public int getCount() {
        return mData == null ? 0 : mData.size();
    }

    @Override
    public T getItem(int position) {
        return mData.get(position);
    }

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

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        //實例化一個ViewHolder
        ViewHolder holder = getViewHolder(position, convertView, parent);
        //使用對外公開的convert方法,通過ViewHolder把View找到,通過Item設置值
        convert(holder, getItem(position));
        return holder.getmConvertView();
    }

    private ViewHolder getViewHolder(int position, View convertView, ViewGroup parent){
        return ViewHolder.getHolder(mContext, mItemLayoutId, convertView, parent);
    }

    /**
     * 對外公佈瞭一個convert方法,並且還把ViewHolder和本item對應的Bean對象給傳出去
     * 現在convert方法裡面需要幹嘛呢?通過ViewHolder把View找到,通過Item設置值
     */
    public abstract void convert(ViewHolder holder, T item);

}

5、item_string_listview.xml

 




    

    

    

    

運行結果如圖:

 

發佈留言

發佈留言必須填寫的電子郵件地址不會公開。 必填欄位標示為 *