上一篇文章简单介绍了下ListView 的优化
可是大家发现,一个优化后的代码却变得非常复杂了,多了一个ViewHolder的东西,写起来更加麻烦了。那么这篇文章就是为了解决这个问题而写的。
先来封装一个通用的ViewHolder
如何封装ViewHolder呢?
上一篇文章我说过,ViewHolder其实就是一个普通的类,这个类保存了我每个Item使用的控件的引用,避免每次都要调用findViewById方法去获取数据。
那么我们需要创建一个通用的ViewHolder类来保存所有的控件的引用,还要方便的查找到这个控件对吧。
所以这个地方我们可以使用HashMap保存控件了,
大概是这样的
public class BaseViewHolder{
private final HashMap<View> mViews;
private View mConvertView;
}
mViews里面可以保存各种控件,这样就方便后面的使用了,不过Android提供了一个比map更适合的数据结构SparseArray,这个效率在这个地方使用比HashMap更好一些,所以大概就是这样了。
public class BaseViewHolder{
private final SparseArray<View> mViews;
private View mConvertView;
}
这个只是创建了一个保存引用的集合,那么具体如何使用才是我们要讲的重点了。
我们根据现在的情况重新封装一下 Adapter
package com.iscoding.lib.base.adapter;
import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import java.util.List;
/**
* 类功能:TODO
* 公司:IsCoding工作室
* 作者:IsCoding
* 邮箱:iscoding@126.com QQ:1400100300
*/
public class TestAdapter<T> extends BaseListAdapter<T> {
private int layoutId;
public TestAdapter(Context context, List<T> list,int layoutId) {
super(context, list);
this.layoutId = layoutId;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if(convertView == null){
convertView = mLayoutInflater.inflate(layoutId,null);
}
BaseViewHolder viewHolder = (BaseViewHolder) convertView.getTag();
if(viewHolder == null){
viewHolder = new BaseViewHolder();
convertView.setTag(viewHolder);
}
// viewHolder.tv_simple.setText(mList.get(position));
return convertView;
}
}
大概是这样了,过去创建好 ViewHolder 就要开始初始化控件了,可是现在呢,我们初始化不了,为什么呢?因为封装的需求,我们目前不知道具体有什么控件,更不知道控件的id是什么,那么我就要封装个方法在这个控件使用的时候把他保存起来,下次使用直接获取这样就达到我们的目标了。那就简单的封装一个方法。
完整的ViewHolder 基本就变成了这个样子
package com.iscoding.lib.base.adapter;
import android.util.SparseArray;
import android.view.View;
/**
* 类功能:TODO
* 公司:IsCoding工作室
* 作者:IsCoding
* 邮箱:iscoding@126.com QQ:1400100300
*/
public class BaseViewHolder {
private final SparseArray<View> mViews = new SparseArray<>();
private View mConvertView;
public BaseViewHolder(View mConvertView){
this.mConvertView= mConvertView;
}
public <T extends View> T getViewById(int viewId) {
//先从SparseArray里面去 跟从map 取是一个道理的
View view = mViews.get(viewId);
if (view == null) {
// 如果没有就用 ConvertView 获取这个view,然后添加到SparseArray里面,下次就可以直接取了
view = mConvertView.findViewById(viewId);
mViews.put(viewId, view);
}
return (T) view;
}
}
那么Adapter也重新封装下,现在的样子大概是这样的
package com.iscoding.lib.base.adapter;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import java.util.List;
/**
* 类功能:TODO
* 公司:IsCoding工作室
* 作者:IsCoding
* 邮箱:iscoding@126.com QQ:1400100300
*/
public abstract class BaseListAdapter<T> extends BaseAdapter {
protected Context mContext;
protected List<T> mList ;
protected LayoutInflater mLayoutInflater;
public BaseListAdapter(Context context, List<T> list) {
this.mContext = context;
this.mList = list;
mLayoutInflater = LayoutInflater.from(mContext);
}
@Override
public int getCount() {
return mList.size();
}
@Override
public Object getItem(int position) {
return mList.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
// @Override
// public abstract View getView(int position, View convertView, ViewGroup parent) ;
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if(convertView == null){
convertView = mLayoutInflater.inflate(getLayoutId(),null);
}
BaseViewHolder viewHolder = (BaseViewHolder) convertView.getTag();
if(viewHolder == null){
viewHolder = new BaseViewHolder(convertView);
convertView.setTag(viewHolder);
}
// 上面这些代码其实每个都一样,那么我就把这些留着父类里面,去封装子类
getView(position,viewHolder );
// viewHolder.tv_simple.setText(mList.get(position));
return convertView;
}
public abstract int getLayoutId() ;
public abstract void getView(int position, BaseViewHolder viewHolder ) ;
}
}
那么如何使用新封装的带有BaseViewHolder 的Adapter呢
看看代码吧
public class StringAdapter extends BaseListAdapter<String> {
public StringAdapter(Context context, List<String> list) {
super(context, list);
}
@Override
public int getLayoutId() {
return R.layout.item_simple;
}
@Override
public void getView(int position, BaseViewHolder viewHolder) {
TextView textView = viewHolder.getViewById(R.id.tv_simple);
textView.setText(mList.get(position));
}
}