Android优化ListView实践

在看了一些vogella的文章之后,发现关于android listview性能优化这一段很有意思,于是实践了一下,经过优化,性能确实提升不少!

成都创新互联公司-专业网站定制、快速模板网站建设、高性价比红桥网站开发、企业建站全套包干低至880元,成熟完善的模板库,直接使用。一站式红桥网站制作公司更省心,省钱,快速模板网站建设找我们,业务覆盖红桥地区。费用合理售后完善,十载实体公司更值得信赖。

先看看优化前和优化后的比较:

优化前的log截图:

优化后的log截图:

并且,在不停滚动ListView的过程中,优化之前会出现ANR现象,在AVD上特别容易复现:

然后,优化后显得很流畅,附上对于的log截图:

下面附上相关代码分析:

ListView中的每一个Item由一个ImageView 和一个TextView组成

Layout:

 
 
 
  1.  
  2. android:layout_width="fill_parent" 
  3. android:layout_height="fill_parent" 
  4. android:orientation="horizontal" > 
  5. android:layout_width="wrap_content" 
  6. android:layout_height="fill_parent" />" 
  7. android:layout_width="wrap_content" 
  8. android:layout_height="fill_parent" 
  9. android:layout_marginLeft="15dp" 
  10. android:gravity="center_vertical" /> 
  11.  

Activity继承自ListActivity,我故意增加了Item,方便测试,效果更明显:

 
 
 
  1. public class ListViewDemo extends ListActivity{ 
  2. private final String[] mItems = new String[] { "Android", "iPhone", 
  3. "WindowsMobile", "Blackberry", "WebOS", "Ubuntu", "Windows7", 
  4. "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", 
  5. "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", 
  6. "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", 
  7. "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", 
  8. "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", 
  9. "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", 
  10. "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", 
  11. "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", 
  12.  "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", 
  13. "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", 
  14. "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", 
  15. "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", 
  16. "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", 
  17.  "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", 
  18. "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", 
  19. "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", 
  20. "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", 
  21.  "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", 
  22.  "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", 
  23. "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", 
  24. "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", 
  25. "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", 
  26. "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", 
  27. "Windows7", "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", 
  28. "Max OS X", "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", 
  29. "Linux", "OS/2", "Ubuntu", "Windows7", "Max OS X", "Linux", "OS/2" }; 
  30.  @Override 
  31. public void onCreate(Bundle savedInstanceState) { 
  32.     super.onCreate(savedInstanceState); 
  33.      ListViewArrayAdapter adapter = new ListViewArrayAdapter(this, mItems); 
  34.       getListView().setAdapter(adapter); 
  35.     } 

然后custom Adapter,优化之前的adapter:

 
 
 
  1. @Override 
  2.       public View getView(int position, View convertView, ViewGroup parent) { 
  3.         long start = System.currentTimeMillis(); 
  4.         LayoutInflater inflater = (LayoutInflater) mContext.getLayoutInflater(); 
  5.         View rowView = inflater.inflate(mViewResourceId, parent, false); 
  6.         TextView textView = (TextView) rowView 
  7.                 .findViewById(mTextViewResourceId); 
  8.         ImageView imageView = (ImageView) rowView 
  9.                 .findViewById(mImageViewResourceId); 
  10.         textView.setText(mNames[position]); 
  11.         String s = mNames[position]; 
  12.         if (s.startsWith("Windows7") || s.startsWith("iPhone")) { 
  13.             imageView.setImageResource(R.drawable.no); 
  14.         } else { 
  15.             imageView.setImageResource(R.drawable.yes); 
  16.         } 
  17.               
  18.        Log.v("jerikc","cost time = " + (System.currentTimeMillis() - start)); 
  19.       return rowView; 

优化之后的Adapter:

 
 
 
  1. public class ListViewArrayAdapter extends ArrayAdapter
  2.        private final Activity mContext; 
  3.        private final String[] mNames; 
  4.        private final static int mViewResourceId = R.layout.text_image_row_layout; 
  5.        private final static int mTextViewResourceId = R.id.textView; 
  6.        private final static int mImageViewResourceId = R.id.imageView; 
  7.     static class ViewHolder { 
  8.         public TextView text; 
  9.         public ImageView image; 
  10.     }     
  11.        public ListViewArrayAdapter(Activity context, String[] names) { 
  12.         super(context, mViewResourceId, names);  
  13.         this.mContext = context; 
  14.         this.mNames = names; 
  15.         } 
  16. @Override 
  17.     public View getView(int position, View convertView, ViewGroup parent) { 
  18.        long start = System.currentTimeMillis(); 
  19.         View rowView = convertView; 
  20.      if (rowView == null) { 
  21.          LayoutInflater inflater = mContext.getLayoutInflater(); 
  22.           rowView = inflater.inflate(mViewResourceId, null); 
  23.          ViewHolder viewHolder = new ViewHolder(); 
  24.         viewHolder.text = (TextView) rowView.findViewById(mTextViewResourceId); 
  25.         viewHolder.image = (ImageView) rowView.findViewById(mImageViewResourceId); 
  26.          rowView.setTag(viewHolder); 
  27.         } 
  28.        ViewHolder holder = (ViewHolder) rowView.getTag(); 
  29.        String s = mNames[position]; 
  30.        holder.text.setText(s); 
  31.       if (s.startsWith("Windows7") || s.startsWith("iPhone")) { 
  32.           holder.image.setImageResource(R.drawable.no); 
  33.         } else { 
  34.            holder.image.setImageResource(R.drawable.yes); 
  35.        } 
  36.        Log.v("jerikc","cost time = " + (System.currentTimeMillis() - start)); 
  37.         return rowView; 
  38.     } 

优化的大致思想就是:优化之前,每次加载item的时候,都要加载一下布局文件,然后生成一个新的row View对象,然后通过View找到对应的ImageView和TextView,正如我们所知道的那样,加载布局文件时很耗时的,特别是在操作比较频繁 情况下,这是不可忍受的,所以会导致ANR现象。

因此,我们可以重复利用已不可见的row View对象。Android中,当它决定让row View对象不可见的时候,它允许通过getView方法中的convertView参数来重复利用刚刚不可见的row View对象。

在优化的过程中,第一次加载的时候,我们需要把相关的数据保存起来,而View有一个方法setTag,该方法可用来保存一些数据结构。我们一个row View对象是由ImageView和TextView空间组成的,因此定义一个ViewHolder来保存ImageView和TextView对象。 在重复利用的过程中,只需简单修改它们的值,而不用再次findViewById。

网站栏目:Android优化ListView实践
文章URL:http://www.csdahua.cn/qtweb/news49/538549.html

网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网