Adapter中getView函数的了解

Adapter中getView函数的理解
      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;      
}


       程序一样很简单,同时效率不受影响,重复代码也少。