整理自: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已经自己实现了,抓住主要脉络思想 实现起来还是不难的。