ym——Android之ListView性能优化

若是有看过我写过的15k面试题的朋友们必定知道,ListView的优化方式有如下几种: java

  1. 重用了convertView
  2. ViewHolder
  3. static class ViewHolder
  4. 在列表里面有图片的状况下,监听滑动不加载图片
以上是大体的说了如下,应付面试已经足够了,若是要使用到项目中,可能有些初学者就迷茫了。接下来我详细的说一下,这个是如何优化的。
   重用了convertView
getView这个方法会调用的次数是大家的数据条目数*2, 重用了convertView,很大程度上的减小了内存的消耗。经过判断convertView是否为null,是的话就须要产生一个视图出来,而后给这个视图数据,最后将这个视图返回给底层,呈献给用户。 
特色 :若是当前的convertView为null,则经过LayoutInflat产生一个view。 
[java]  view plain copy print ?
  1. <span style="font-size:14px;">public View getView(int position,View convertView,ViewGroup parent)   
  2. {   
  3. if(convertView==null)   
  4. {   
  5. convertView=LayoutInflater.from(context).inflate(R.layout.section_list_item1,null);   
  6. }   
  7. TextViewtv_name=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_name);   
  8. TextViewtv_phone=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_phoneNum);  
  9. ContactInfo1confo=contacts.get(position);   
  10. if(confo!=null){//toseteveryitem'stext   
  11. tv_name.setText(confo.getContactName());   
  12. tv_phone.setText(confo.getContact_Phone());   
  13. }   
  14. return convertView;   
  15. }</span><span style="font-size: 24px;"> </span>  


ViewHolder
上面的写法会有一个缺点,就是每次在getVIew的时候,都须要从新的findViewById,从新找到控件,而后进行控件的赋值以及事件相应设置。这样其实在作重复的事情,由于的geiview中,其实包含有这些控件,并且这些控件的id还都是同样的,也就是其实只要在view中findViewById一次,后面无须要每次都要findViewById了。 
下面给出第二种写法 
写发的特色,一般有一个内部类classViewHolder,这个ViewHolder,用来标识view中一些控件,方便进行一些事件相应操做的设置,好比onClick等等,这样能够不用每次都要findViewById了,减小了性能的消耗。同时重用了convertView,很大程度上的减小了内存的消耗。 
[java]  view plain copy print ?
  1. <span style="font-size:14px;">public View getView(int position,View convertView,ViewGroup parent)   
  2. {   
  3. ViewHolderholder;   
  4. if(convertView==null){   
  5. convertView=LayoutInflater.from(context).inflate(R.layout.section_list_item1,null);   
  6. holder=newViewHolder();   
  7. holder.tv_name=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_name);   
  8. holder.tv_phone=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_phoneNum);   
  9. convertView.setTag(holder);   
  10. }   
  11. else   
  12. {   
  13. holder=(ViewHolder)convertView.getTag();   
  14. }   
  15. ContactInfo1confo=contacts.get(position);   
  16. Log.i("my","confo"+confo.getContactName());   
  17. if(confo!=null){//toseteveryitem'stext   
  18.   
  19. holder.tv_name.setText(confo.getContactName());   
  20. holder.tv_phone.setText(confo.getContact_Phone());   
  21. }   
  22. return convertView;   
  23. }   
  24. classViewHolder   
  25. {   
  26. TextViewtv_name,tv_phone;   
  27. } </span>  


static class ViewHolder
把以上两种结合起来,而后把 ViewHolder为static,也就是静态的,静态类只会在第一次加载时会耗费比较长时间,可是后面就能够很好帮助加载,
同时保证了内存中只有一个ViewHolder,节省了内存的开销。 
咱们会发现不管是什么adapter都是这些优化的方式,因此你们确定思考会没有一种方法能够把代码重用的部分抽出来,
若是有这种想法的朋友能够看下我写的这篇adapter代码优化
在列表里面有图片的状况下,监听滑动不加载图片
还有一种状况,就是一个列表加载不少图片,在图片还没加载下来的时候,快速的滑动列表,会发现卡顿的现象发生。
这种状况咱们就须要监听,ListView的滑动监听里面设置不加载图片,滑动中止开始加载。
[java]  view plain copy print ?
  1. 1、判断listView状态  
  2. AbsListView.OnScrollListener onScrollListener = new AbsListView.OnScrollListener() {// ListView  
  3. // 触摸事件  
  4.   
  5. public void onScroll(AbsListView view, int firstVisibleItem,  
  6. int visibleItemCount, int totalItemCount) {  
  7. }  
  8.   
  9. public void onScrollStateChanged(AbsListView view, int scrollState) {  
  10. switch (scrollState) {  
  11. case AbsListView.OnScrollListener.SCROLL_STATE_FLING:// 滑动状态  
  12. threadFlag = false;  
  13. break;  
  14. case AbsListView.OnScrollListener.SCROLL_STATE_IDLE:// 中止  
  15. threadFlag = true;  
  16. startThread();  
  17. break;  
  18. case AbsListView.OnScrollListener.SCROLL_STATE_TOUCH_SCROLL:// 触摸listView  
  19. threadFlag = false;  
  20. break;  
  21. default:  
  22. // Toast.makeText(contextt, "default",  
  23. // Toast.LENGTH_SHORT).show();  
  24. break;  
  25. }  
  26. }  
  27. };  
  28. 2、获取当前屏幕上显示的items:  
  29. mListView.getFirstVisiblePosition();  
相关文章
相关标签/搜索