ListView控件在Android应用程序中被使用的频率非常之高,而ListView的主要代码都集中在Adapter中,因为ListView每一行的子View的产生与数据填充都是Adapter中做的,其实就是getView回调函数。getView函数非常重要,技术面试官也喜欢问应聘者对该函数的理解,尤其是View的重用。
getView函数的完整定义如下:
public View getView(int position, View convertView, ViewGroup parent)
其中返回值就是返回给ListView每一行的子View
position为ListView中的第几行
convertView为之前由该函数返回的View,系统再传入进来供我们复用
parent为父容器,一般可以不用到
开发起来其实非常简单,用position来找数据(在数据数组或List中利用position找),convertView为空则利用xml产生一个convertView并返回,不为空则直接返回之,当然返回之前要做的就是把数据填入到convertView中。
其实这样就可以了,填入数据可以用convertView.findViewById函数去找子控件然后设置值。典型代码如下:
public View getView(int position, View convertView, ViewGroup parent) {
if ( convertView == null) {
convertView = inflater.inflate(R.layout.listview_item, null);
}
MyObj myObj = dataList.get(position);
((TextView)convertView.findViewById(R.id.textView1)).setText(myObj.textView1Data);
......
return convertView;
}
代码很简单,但是我们却很少这样写,因为许多人会用到一个内部类,一般取名为ViewHolder,结构一般为:
private class ViewHolder {
public TextView textView1;
......
}
使用的时候一般是这样的:
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if ( convertView == null) {
holder = new ViewHolder();
convertView = inflater.inflate(R.layout.listview_item, null);
holder.textView1 = (TextView)convertView.findViewById(R.id.textView1);
......
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
}
MyObj myObj = dataList.get(position);
holder.textView1.setText(myObj.textView1Data);
......
return convertView;
}
本来可以写得很简单,加入ViewHolder后好像变复杂了,而且许多人认为ViewHolder是必配的,其实通过前面更原始的写法就可以判断ViewHolder不是必须的,用的理由就是一个:提高效率。
因为ViewHolder对象里持有当前convertView中每一个子View的引用,找子View用ViewHolder对象直接访问比用findViewById来的更快。所以ViewHolder内部类唯一的作用就是缓存子View,让程序找起来更快而已。
通过前面的分析,既然ViewHolder不是必须的,只用来缓存,而且我们程序中有许多的Adapter,如果每个Adapter都定义一个内部类ViewHolder,显然会有很多重复代码。那么提炼一个公共方法专门缓存子View就很容易被想到了。下面是别人提炼出来的公共方法:
public class ViewHolder {
public static <T extends View> T get(View view, int id) {
SparseArray<View> viewHolder = (SparseArray<View>) view.getTag();
if (viewHolder == null) {
viewHolder = new SparseArray<View>();
view.setTag(viewHolder);
}
View childView = viewHolder.get(id);
if (childView == null) {
childView = view.findViewById(id);
viewHolder.put(id, childView);
}
return (T) childView;
}
}
具体使用如下:
public View getView(int position, View convertView, ViewGroup parent) {
if ( convertView == null) {
convertView = inflater.inflate(R.layout.listview_item, null);
}
MyObj myObj = dataList.get(position);
TextView textView1 = ViewHolder.get(convertView,R.id.textView1)
textView1.setText(myObj.textView1Data);
......
return convertView;
}
程序一样很简单,同时效率不受影响,重复代码也少。