Android 快速開發系列 打造萬能的ListView GridView 適配器
來源:程序員人生 發布時間:2014-09-09 17:30:45 閱讀次數:2816次
轉載請標明出處:http://blog.csdn.net/lmj623565791/article/details/38902805 ,本文出自【張鴻洋的博客】
1、概述
相信做Android開發的寫得最多的就是ListView,GridView的適配器吧,記得以前開發一同事開發項目,一個項目下來基本就一直在寫ListView的Adapter都快吐了~~~對于Adapter一般都繼承BaseAdapter復寫幾個方法,getView里面使用ViewHolder模式,其實大部分的代碼基本都是類似的。
本篇博客為快速開發系列的第一篇,將一步一步帶您封裝出一個通用的Adapter。
2、常見的例子
首先看一個最常見的案例,大家一目十行的掃一眼
1、布局文件
主布局文件:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<ListView
android:id="@+id/id_lv_main"
android:layout_width="fill_parent"
android:layout_height="fill_parent" />
</RelativeLayout>
Item的布局文件:
<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/id_tv_title"
android:layout_width="match_parent"
android:layout_height="50dp"
android:background="#aa111111"
android:gravity="center_vertical"
android:paddingLeft="15dp"
android:textColor="#ffffff"
android:text="hello"
android:textSize="20sp"
android:textStyle="bold" >
</TextView>
2、Adapter
package com.example.zhy_baseadapterhelper;
import java.util.List;
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 LayoutInflater mInflater;
private Context mContext;
private List<String> mDatas;
public MyAdapter(Context context, List<String> mDatas)
{
mInflater = LayoutInflater.from(context);
this.mContext = context;
this.mDatas = mDatas;
}
@Override
public int getCount()
{
return mDatas.size();
}
@Override
public Object 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)
{
ViewHolder viewHolder = null;
if (convertView == null)
{
convertView = mInflater.inflate(R.layout.item_single_str, parent,
false);
viewHolder = new ViewHolder();
viewHolder.mTextView = (TextView) convertView
.findViewById(R.id.id_tv_title);
convertView.setTag(viewHolder);
} else
{
viewHolder = (ViewHolder) convertView.getTag();
}
viewHolder.mTextView.setText(mDatas.get(position));
return convertView;
}
private final class ViewHolder
{
TextView mTextView;
}
}
3、Activity
package com.example.zhy_baseadapterhelper;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;
public class MainActivity extends Activity
{
private ListView mListView;
private List<String> mDatas = new ArrayList<String>(Arrays.asList("Hello",
"World", "Welcome"));
private MyAdapter mAdapter;
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mListView = (ListView) findViewById(R.id.id_lv_main);
mListView.setAdapter(mAdapter = new MyAdapter(this, mDatas));
}
}
上面這個例子大家應該都寫了無數遍了,MyAdapter集成BaseAdapter,然后getView里面使用ViewHolder模式;一般情況下,我們的寫法是這樣的:對于不同布局的ListView,我們會有一個對應的Adapter,在Adapter中又會有一個ViewHolder類來提高效率。
這樣出現ListView就會出現與之對于的Adapter類、ViewHolder類;那么有沒有辦法減少我們的編碼呢?
下面首先拿ViewHolder開刀~
3、通用的ViewHolder
首先分析下ViewHolder的作用,通過convertView.setTag與convertView進行綁定,然后當convertView復用時,直接從與之對于的ViewHolder(getTag)中拿到convertView布局中的控件,省去了findViewById的時間~
也就是說,實際上們每個convertView會綁定一個ViewHolder對象,這個viewHolder主要用于幫convertView存儲布局中的控件。
那么我們只要寫出一個通用的ViewHolder,然后對于任意的convertView,提供一個對象讓其setTag即可;
既然是通用,那么我們這個ViewHolder就不可能含有各種控件的成員變量了,因為每個Item的布局是不同的,最好的方式是什么呢?
提供一個容器,專門存每個Item布局中的所有控件,而且還要能夠查找出來;既然需要查找,那么ListView肯定是不行了,需要一個鍵值對進行保存,鍵為控件的Id,值為控件的引用,相信大家立刻就能想到Map;但是我們不用Map,因為有更好的替代類,就是我們android提供的SparseArray這個類,和Map類似,但是比Map效率,不過鍵只能為Integer.
下面看我們的ViewHolder類:
package com.example.zhy_baseadapterhelper;
import android.content.Context;
import android.util.Log;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
public class ViewHolder
{
private final SparseArray<View> mViews;
private View mConvertView;
private ViewHolder(Context context, ViewGroup parent, int layoutId,
int position)
{
this.mViews = new SparseArray<View>();
mConvertView = LayoutInflater.from(context).inflate(layoutId, parent,
false);
//setTag
mConvertView.setTag(this);
}
/**
* 拿到一個ViewHolder對象
* @param context
* @param convertView
* @param parent
* @param layoutId
* @param position
* @return
*/
public static ViewHolder get(Context context, View convertView,
ViewGroup parent, int layoutId, int position)
{
if (convertView == null)
{
return new ViewHolder(context, parent, layoutId, position);
}
return (ViewHolder) convertView.getTag();
}
/**
* 通過控件的Id獲取對于的控件,如果沒有則加入views
* @param viewId
* @return
*/
public <T extends View> T getView(int viewId)
{
View view = mViews.get(viewId);
if (view == null)
{
view = mConvertView.findViewById(viewId);
mViews.put(viewId, view);
}
return (T) view;
}
public View getConvertView()
{
return mConvertView;
}
}
與傳統的ViewHolder不同,我們使用了一個SparseArray<View>用于存儲與之對于的convertView的所有的控件,當需要拿這些控件時,通過getView(id)進行獲取;
下面看使用該ViewHolder的MyAdapter;
@Override
public View getView(int position, View convertView, ViewGroup parent)
{
//實例化一個viewHolder
ViewHolder viewHolder = ViewHolder.get(mContext, convertView, parent,
R.layout.item_single_str, position);
//通過getView獲取控件
TextView tv = viewHolder.getView(R.id.id_tv_title);
//使用
tv.setText(mDatas.get(position));
return viewHolder.getConvertView();
}
只看getView,其他方法都一樣;首先調用ViewHolder的get方法,如果convertView為null,new一個ViewHolder實例,通過使用mInflater.inflate加載布局,然后new一個SparseArray用于存儲View,最后setTag(this);
如果存在那么直接getTag
最后通過getView(id)獲取控件,如果存在則直接返回,否則調用findViewById,返回存儲,返回。
好了,一個通用的ViewHolder寫好了,以后一個項目幾十個Adapter一個ViewHolder直接hold住全場~~大家可以省點時間斗個小地主了~~
4、打造通用的Adapter
有了通用的ViewHolder大家肯定不能滿足,怎么也得省出dota的時間,人在塔在~~
下面看如何打造一個通過的Adapter,我們叫做CommonAdapter
繼續分析,Adapter一般需要保持一個List對象,存儲一個Bean的集合,不同的ListView,Bean肯定是不同的,這個CommonAdapter肯定需要支持泛型,內部維持一個List<T>,就解決我們的問題了;
于是我們初步打造我們的CommonAdapter
package com.example.zhy_baseadapterhelper;
import java.util.List;
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 abstract class CommonAdapter<T> extends BaseAdapter
{
protected LayoutInflater mInflater;
protected Context mContext;
protected List<T> mDatas;
public CommonAdapter(Context context, List<T> mDatas)
{
mInflater = LayoutInflater.from(context);
this.mContext = context;
this.mDatas = mDatas;
}
@Override
public int getCount()
{
return mDatas.size();
}
@Override
public Object getItem(int position)
{
return mDatas.get(position);
}
@Override
public long getItemId(int position)
{
return position;
}
}
我們的CommonAdapter依然是一個抽象類,除了getView以外我們把其他的代碼都實現了,這樣的話,在使用我們的Adapter只要實現一個getView,然后getView里面再使用我們打造的通過的ViewHolder是不是感覺還不錯~
現在我們的MyAdapter是這樣的:
package com.example.zhy_baseadapterhelper;
import java.util.List;
import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
public class MyAdapter<T> extends CommonAdapter<T>
{
public MyAdapter(Context context, List<T> mDatas)
{
super(context, mDatas);
}
@Override
public View getView(int position, View convertView, ViewGroup parent)
{
ViewHolder viewHolder = ViewHolder.get(mContext, convertView, parent,
R.layout.item_single_str, position);
TextView mTitle = viewHolder.getView(R.id.id_tv_title);
mTitle.setText((String) mDatas.get(position));
return viewHolder.getConvertView();
}
}
所有的代碼加起來也就10行左右,是不是神清氣爽~~稍等,我先去dota一把~
但是我們是否就這樣滿足了呢?顯然還可以簡化。
5、進一步鑄造
注意我們的getView里面的代碼,雖然只有4行,但是我覺得所有的Adapter的
第一行(ViewHolder viewHolder = getViewHolder(position, convertView,parent);)和
最后一行:return viewHolder.getConvertView();一定是一樣的。
那么我們可以這樣做:我們把第一行和最后一行寫死,把中間變化的部分抽取出來,這不就是OO的設計原則嘛。現在CommonAdapter是這樣的:
package com.example.zhy_baseadapterhelper;
import java.util.List;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
public abstract class n<T> extends BaseAdapter
{
protected LayoutInflater mInflater;
protected Context mContext;
protected List<T> mDatas;
protected final int mItemLayoutId;
public CommonAdapter(Context context, List<T> mDatas, int itemLayoutId)
{
this.mContext = context;
this.mInflater = LayoutInflater.from(mContext);
this.mDatas = mDatas;
this.mItemLayoutId = itemLayoutId;
}
@Override
public int getCount()
{
return mDatas.size();
}
@Override
public T 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)
{
final ViewHolder viewHolder = getViewHolder(position, convertView,
parent);
convert(viewHolder, getItem(position));
return viewHolder.getConvertView();
}
public abstract void convert(ViewHolder helper, T item);
private ViewHolder getViewHolder(int position, View convertView,
ViewGroup parent)
{
return ViewHolder.get(mContext, convertView, parent, mItemLayoutId,
position);
}
}
對外公布了一個convert方法,并且還把viewHolder和本Item對于的Bean對象給傳出去,現在convert方法里面需要干嘛呢?
通過ViewHolder把View找到,通過Item設置值;
現在我覺得代碼簡化到這樣,我已經不需要單獨寫一個Adapter了,直接MainActivity匿名內部類走起~
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mListView = (ListView) findViewById(R.id.id_lv_main);
//設置適配器
mListView.setAdapter(mAdapter = new CommonAdapter<String>(
getApplicationContext(), mDatas, R.layout.item_single_str)
{
@Override
public void convert(ViewHolder c, String item)
{
TextView view = viewHolder.getView(R.id.id_tv_title);
view.setText(item);
}
});
}
可以看到效果咋樣,不錯吧。你覺得還能簡化么?我覺得還能改善。
6、Adapter最后的封魔
我們現在在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);
那么現在ViewHolder是:
package com.example.zhy_baseadapterhelper;
import android.content.Context;
import android.graphics.Bitmap;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;
import com.example.zhy_baseadapterhelper.ImageLoader.Type;
public class ViewHolder
{
private final SparseArray<View> mViews;
private int mPosition;
private View mConvertView;
private ViewHolder(Context context, ViewGroup parent, int layoutId,
int position)
{
this.mPosition = position;
this.mViews = new SparseArray<View>();
mConvertView = LayoutInflater.from(context).inflate(layoutId, parent,
false);
// setTag
mConvertView.setTag(this);
}
/**
* 拿到一個ViewHolder對象
*
* @param context
* @param convertView
* @param parent
* @param layoutId
* @param position
* @return
*/
public static ViewHolder get(Context context, View convertView,
ViewGroup parent, int layoutId, int position)
{
if (convertView == null)
{
return new ViewHolder(context, parent, layoutId, position);
}
return (ViewHolder) convertView.getTag();
}
public View getConvertView()
{
return mConvertView;
}
/**
* 通過控件的Id獲取對于的控件,如果沒有則加入views
*
* @param viewId
* @return
*/
public <T extends View> T getView(int viewId)
{
View view = mViews.get(viewId);
if (view == null)
{
view = mConvertView.findViewById(viewId);
mViews.put(viewId, view);
}
return (T) view;
}
/**
* 為TextView設置字符串
*
* @param viewId
* @param text
* @return
*/
public ViewHolder setText(int viewId, String text)
{
TextView view = getView(viewId);
view.setText(text);
return this;
}
/**
* 為ImageView設置圖片
*
* @param viewId
* @param drawableId
* @return
*/
public ViewHolder setImageResource(int viewId, int drawableId)
{
ImageView view = getView(viewId);
view.setImageResource(drawableId);
return this;
}
/**
* 為ImageView設置圖片
*
* @param viewId
* @param drawableId
* @return
*/
public ViewHolder setImageBitmap(int viewId, Bitmap bm)
{
ImageView view = getView(viewId);
view.setImageBitmap(bm);
return this;
}
/**
* 為ImageView設置圖片
*
* @param viewId
* @param drawableId
* @return
*/
public ViewHolder setImageByUrl(int viewId, String url)
{
ImageLoader.getInstance(3, Type.LIFO).loadImage(url,
(ImageView) getView(viewId));
return this;
}
public int getPosition()
{
return mPosition;
}
}
現在的MainActivity只需要這么寫:
mAdapter = new CommonAdapter<String>(getApplicationContext(),
R.layout.item_single_str, mDatas)
{
@Override
protected void convert(ViewHolder viewHolder, String item)
{
viewHolder.setText(R.id.id_tv_title, item);
}
};
convertView里面只要一行代碼了~~~
好了,到此我們的通用的Adapter已經一步一步鑄造完畢~咋樣,以后寫項目省下來的時間是不是可以陪我切磋dota了(ps:11昵稱:血魔哥404)~~
注:關于ViewHolder里面的setText,setImageResource這類的方法,大家可以在使用的過程中不斷的完善,今天發現這個控件可以這么設置值,好,放進去;時間長了,基本就完善了。還有那個ImageLoader是我另一篇博客里的,大家可以使用UIL,Volley或者自己寫個圖片加載器;
7、實踐
說了這么多,還是得拿出來讓我們的實踐檢驗檢驗,順便來幾張套圖,俗話說,沒圖沒正相。
1、我們的實例代碼的圖是這樣的:

關于Adapter和ViewHolder的代碼是這樣的:
// 設置適配器
mListView.setAdapter(mAdapter = new CommonAdapter<String>(
getApplicationContext(), mDatas, R.layout.item_single_str)
{
@Override
public void convert(ViewHolder helper, String item)
{
helper.setText(R.id.id_tv_title,item);
}
});
哎喲,我是不是只要貼一行;
2、來個復雜點的布局
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="#ffffff"
android:orientation="vertical"
android:padding="10dp" >
<TextView
android:id="@+id/tv_title"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:singleLine="true"
android:text="紅色錢包"
android:textSize="16sp"
android:textColor="#444444" >
</TextView>
<TextView
android:id="@+id/tv_describe"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_below="@id/tv_title"
android:layout_marginTop="10dp"
android:maxLines="2"
android:minLines="1"
android:text="周三早上丟失了紅色錢包,在食堂二樓"
android:textColor="#898989"
android:textSize="16sp" >
</TextView>
<TextView
android:id="@+id/tv_time"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/tv_describe"
android:layout_marginTop="10dp"
android:text="20130240122"
android:textColor="#898989"
android:textSize="12sp" >
</TextView>
<TextView
android:id="@+id/tv_phone"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentRight="true"
android:layout_below="@id/tv_describe"
android:layout_marginTop="10dp"
android:background="#5cbe6c"
android:drawableLeft="@drawable/icon_photo"
android:drawablePadding="5dp"
android:paddingBottom="3dp"
android:paddingLeft="5dp"
android:paddingRight="5dp"
android:paddingTop="3dp"
android:text="138024249542"
android:textColor="#ffffff"
android:textSize="12sp" >
</TextView>
</RelativeLayout>
效果圖是這樣的:

布局是不是挺復雜的了~~
但是代碼是這樣的:
// 設置適配器
mListView.setAdapter(mAdapter = new CommonAdapter<Bean>(
getApplicationContext(), mDatas, R.layout.item_list)
{
@Override
public void convert(ViewHolder helper, Bean item)
{
helper.setText(R.id.tv_title, item.getTitle());
helper.setText(R.id.tv_describe, item.getDesc());
helper.setText(R.id.tv_phone, item.getPhone());
helper.setText(R.id.tv_time, item.getTime());
// helper.getView(R.id.tv_title).setOnClickListener(l)
}
});
從一個字符串的布局到這樣的布局,Adapter加ViewHolder的改變就這么多,加起來3行左右代碼~~~
到此,Android 快速開發系列 打造萬能的ListView GridView 適配器結束;
最后給大家推薦一個gitHub項目:https://github.com/JoanZapata/base-adapter-helper ,這個項目所做的,和我上面寫的基本一致。
還有上面的布局文件來自網絡,感謝Bmob的提供~
好了,我要去快樂的玩耍了~~
源碼點擊下載
生活不易,碼農辛苦
如果您覺得本網站對您的學習有所幫助,可以手機掃描二維碼進行捐贈