Android ListView优化详解

ListView的Adapter的作用如下图所示:


Adapter的作用就是ListView界面与数据之间的桥梁,当列表里的每一项显示到页面时,都会调用Adapter的getView方法返回一个View。想过没有? 在我们的列表有1000000项时会是什么样的?是不是会占用极大的系统资源?

先看看下面的代码:

Java代码     
  1. public View getView(int position, View convertView, ViewGroup parent) { 
  2.     View item = mInflater.inflate(R.layout.list_item_icon_text, null); 
  3.     ((TextView) item.findViewById(R.id.text)).setText(DATA[position]); 
  4.     ((ImageView) item.findViewById(R.id.icon)).setImageBitmap(R.drawable.icon); 
java代码 
  1. public View getView(int position, View convertView, ViewGroup parent) {
  2. View item = mInflater.inflate(R.layout.list_item_icon_text, null);
  3. ((TextView) item.findViewById(R.id.text)).setText(DATA[position]);
  4. ((ImageView) item.findViewById(R.id.icon)).setImageBitmap(R.drawable.icon);
  5. }

怎么样?如果超过1000000项时,后果不堪设想!您可千万别这么写!

优化方案:

方案一:

Java代码
  1. public View getView(int position, View convertView, ViewGroup parent) { 
  2.      
  3.     if (convertView == null) { 
  4.         convertView = mInflater.inflate(R.layout.item, null); 
  5.     } 
  6.  
  7.     ((TextView) item.findViewById(R.id.text)).setText(DATA[position]); 
  8.     ((ImageView) item.findViewById(R.id.icon)).setImageBitmap(R.drawable.icon); 
java代码 
  1. public View getView(int position, View convertView, ViewGroup parent) {
  2. if (convertView == null) {
  3. convertView = mInflater.inflate(R.layout.item, null);
  4. }
  5. ((TextView) item.findViewById(R.id.text)).setText(DATA[position]);
  6. ((ImageView) item.findViewById(R.id.icon)).setImageBitmap(R.drawable.icon);
  7. }

上面的系统将会减少创建很多View。性能得到了很大的提升。

方案2:

Java代码
  1. public View getView(int position, View convertView, ViewGroup parent) { 
  2.         ViewHolder holder; 
  3.         if (convertView == null) { 
  4.             convertView = mInflater.inflate(R.layout.list_item_icon_text, null); 
  5.             holder = new ViewHolder(); 
  6.             holder.text = (TextView) convertView.findViewById(R.id.text); 
  7.             holder.icon = (ImageView) convertView.findViewById(R.id.icon); 
  8.             convertView.setTag(holder);//给View添加一个格外的数据 
  9.         } else { 
  10.             holder = (ViewHolder) convertView.getTag();//将上面的holder数据取出来 
  11.  
  12.         } 
  13.         holder.text.setText(DATA[position]); 
  14.         holder.icon.setImageBitmap(R.drawable.icon); 
  15.         return convertView; 
  16.     } 
  17.  
  18.     static class ViewHolder { 
  19.         TextView text; 
  20.         ImageView icon; 
  21.     } 
java代码 
  1. public View getView(int position, View convertView, ViewGroup parent) {
  2. ViewHolder holder;
  3. if (convertView == null) {
  4. convertView = mInflater.inflate(R.layout.list_item_icon_text, null);
  5. holder = new ViewHolder();
  6. holder.text = (TextView) convertView.findViewById(R.id.text);
  7. holder.icon = (ImageView) convertView.findViewById(R.id.icon);
  8. convertView.setTag(holder);//给View添加一个格外的数据
  9. else {
  10. holder = (ViewHolder) convertView.getTag();//将上面的holder数据取出来
  11. }
  12. holder.text.setText(DATA[position]);
  13. holder.icon.setImageBitmap(R.drawable.icon);
  14. return convertView;
  15. }
  16. static class ViewHolder {
  17. TextView text;
  18. ImageView icon;
  19. }

怎么样?会不会又给您的系统带来很大的提升呢?看看下面三种方式的性能对比图您就知道了!

  可以发现,只有第一屏(可视范围)调用getView所消耗的时间远远多于后面的,通过对

convertView == null内代码监控也是同样的结果。 也就是说ListView仅仅缓存了可视范围内的View,随后的滚动都是对这些View进行数据更新不管你有多少数据,他都只用ArrayList缓存可视范围内的View,这样保证了性能,也造成了我以为ListView只缓存View结构不缓存数据的假相(不会只有我一人这么认为吧- - #)。这也能解释为什么GOOGLE优化方案一比二高很多的原因。那么剩下的也就只有findViewById比较耗时了。据此大家可以看看AbsListView的 源代码,看看
obtainView这个方法内的代码及RecycleBin这个类的实现,欢迎分享。
Java代码
  1. if (convertView == null) { 
  2. convertView = mInflater.inflate(R.layout.list_item_icon_text, null); 
  3. ((ImageView) convertView.findViewById(R.id.icon1)).setImageResource(R.drawable.icon); 
  4. ((TextView) convertView.findViewById(R.id.text1)).setText(mData[position]); 
  5. ((ImageView) convertView.findViewById(R.id.icon2)).setImageResource(R.drawable.icon); 
  6. ((TextView) convertView.findViewById(R.id.text2)).setText(mData[position]); 
  7.    else 
  8.          return convertView; 
java代码 
  1. if (convertView == null) {
  2. convertView = mInflater.inflate(R.layout.list_item_icon_text, null);
  3. ((ImageView) convertView.findViewById(R.id.icon1)).setImageResource(R.drawable.icon);
  4. ((TextView) convertView.findViewById(R.id.text1)).setText(mData[position]);
  5. ((ImageView) convertView.findViewById(R.id.icon2)).setImageResource(R.drawable.icon);
  6. ((TextView) convertView.findViewById(R.id.text2)).setText(mData[position]);
  7. }
  8. else
  9. return convertView;
没错,你会发现滚动时会重复显示第一屏的数据!

子控件里的事件因为是同一个控件,也可以直接放到convertView == null 代码块内部,如果需要交互数据比如position,可以通过tag方式来设置并获取当前数据。

  这里推荐如果只是一般的应用(一般指子控件不多),无需都是用静态内部类来优化,使用方案1即可;反之,对性能要求较高时可采用。此外需要提醒的是这里也是用空间换时间的做法,View本身因为setTag而会占用更多的内存,还会增加代码量;而findViewById会临时消耗更多的内存,所以不可盲目使用,依实际情况而定。

你可能感兴趣的:(Android ListView优化详解)