整理自:http://blog.csdn.net/lmj623565791/article/details/38902805/
這篇文章只是用來提取上面那個鏈接的重點的,推薦上面的文章看懂之后再看這個
關鍵是2個要點:
- 制造一個通用的adapter,
getItem()
,getItemId()
,getCount()
這些方法的復用很簡單,主要是getView()
方法,這里我們的getView()
方法這么寫:
@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);
}
原本getView的參數全部給我們自己造的ViewHolder來處理,最后返回的convertView也來自我們自己造的ViewHolder。
第二行代碼convert(viewHolder, getItem(position));
還是一個抽象方法,需要我們在用到這個通用Adapter的時候自己實現這個抽象方法,然后這個實現了的方法會在內部被執行(實現之后此時已經不是抽象方法,是可以執行自己設定的邏輯的,這個點之前一直想不清楚),具體convert(viewHolder, getItem(position));
中的實現,我們每次基本都不一樣(添加不同的控件,控件加監聽事件等),注意這里convert()的參數里面也有viewHolder,說明到時候我們實現的邏輯同樣需要用到這個viewHolder
- 接著是關鍵的ViewHolder
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;
}
- 這里的ViewHolder的創建用到了單例模式,如果convertView為null,說明第一次創建,ViewHolder自然也不存在,那就構造ViewHolder構造函數,并用
mConvertView.setTag(this);
保存這個ViewHolder;如果存在,那就直接通過mConvertView.setTag(this);
獲取ViewHolder即可。 - 其中get()方法才是這個的關鍵,傳入
getView()
方法中的所有參數,并傳入了layoutId和Context參數(這兩個參數是實現通用Adapter的時候才會傳入的) - 除此之外還有一個關注的點就是這個ViewHolder存儲View的方式是使用了SparseArray,如果mViews中沒有我們要找的View(通過R.id.···),就把新的View加入到mViews中。
- 初次之外,我們還可以更加簡化代碼,我們在ViewHolder中加入:
/**
* 為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;
}
}
這些很簡單,就不說了。
這個復用adapter已經自己實現了,抓住主要脈絡思想 實現起來還是不難的。