现在的位置: 首页 > 综合 > 正文

ym——Android之ListView性能优化

2017年02月11日 ⁄ 综合 ⁄ 共 3229字 ⁄ 字号 评论关闭

转载请注明本文出自Cym的博客(http://blog.csdn.net/cym492224103),谢谢支持!


Android之ListView性能优化

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

  1. 重用了convertView
  2. ViewHolder
  3. static
    class ViewHolder
  4. 在列表里面有图片的情况下,监听滑动不加载图片
以上是大致的说了以下,应付面试已经足够了,如果要使用到项目中,可能有些初学者就迷茫了。接下来我详细的说一下,这个是如何优化的。
  重用了convertView
getView这个方法会调用的次数是你们的数据条目数*2,重用了convertView,很大程度上的减少了内存的消耗。通过判断convertView是否为null,是的话就需要产生一个视图出来,然后给这个视图数据,最后将这个视图返回给底层,呈献给用户。 

特点:如果当前的convertView为null,则通过LayoutInflat产生一个view。 
<span style="font-size:14px;">public View getView(int position,View convertView,ViewGroup parent) 
{ 
if(convertView==null) 
{ 
convertView=LayoutInflater.from(context).inflate(R.layout.section_list_item1,null); 
} 
TextViewtv_name=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_name); 
TextViewtv_phone=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_phoneNum);
ContactInfo1confo=contacts.get(position); 
if(confo!=null){//toseteveryitem'stext 
tv_name.setText(confo.getContactName()); 
tv_phone.setText(confo.getContact_Phone()); 
} 
return convertView; 
}</span><span style="font-size: 24px;"> </span>
ViewHolder
上面的写法会有一个缺点,就是每次在getVIew的时候,都需要重新的findViewById,重新找到控件,然后进行控件的赋值以及事件相应设置。这样其实在做重复的事情,因为的geiview中,其实包含有这些控件,而且这些控件的id还都是一样的,也就是其实只要在view中findViewById一次,后面无需要每次都要findViewById了。 

下面给出第二种写法 

写发的特点,通常有一个内部类classViewHolder,这个ViewHolder,用来标识view中一些控件,方便进行一些事件相应操作的设置,比如onClick等等,这样可以不用每次都要findViewById了,减少了性能的消耗。同时重用了convertView,很大程度上的减少了内存的消耗。 
<span style="font-size:14px;">public View getView(int position,View convertView,ViewGroup parent) 
{ 
ViewHolderholder; 
if(convertView==null){ 
convertView=LayoutInflater.from(context).inflate(R.layout.section_list_item1,null); 
holder=newViewHolder(); 
holder.tv_name=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_name); 
holder.tv_phone=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_phoneNum); 
convertView.setTag(holder); 
} 
else 
{ 
holder=(ViewHolder)convertView.getTag(); 
} 
ContactInfo1confo=contacts.get(position); 
Log.i("my","confo"+confo.getContactName()); 
if(confo!=null){//toseteveryitem'stext 

holder.tv_name.setText(confo.getContactName()); 
holder.tv_phone.setText(confo.getContact_Phone()); 
} 
return convertView; 
} 
classViewHolder 
{ 
TextViewtv_name,tv_phone; 
} </span>
static class ViewHolder
把以上两种结合起来,然后把ViewHolder为static,也就是静态的,静态类只会在第一次加载时会耗费比较长时间,但是后面就可以很好帮助加载,
同时保证了内存中只有一个ViewHolder,节省了内存的开销。 
我们会发现无论是什么adapter都是这些优化的方式,所以大家肯定思考会没有一种方法可以把代码重用的部分抽出来,
如果有这种想法的朋友可以看下我写的这篇adapter代码优化
在列表里面有图片的情况下,监听滑动不加载图片
还有一种情况,就是一个列表加载很多图片,在图片还没加载下来的时候,快速的滑动列表,会发现卡顿的现象发生。
这种情况我们就需要监听,ListView的滑动监听里面设置不加载图片,滑动停止开始加载。
1、判断listView状态
AbsListView.OnScrollListener onScrollListener = new AbsListView.OnScrollListener() {// ListView
// 触摸事件

public void onScroll(AbsListView view, int firstVisibleItem,
int visibleItemCount, int totalItemCount) {
}

public void onScrollStateChanged(AbsListView view, int scrollState) {
switch (scrollState) {
case AbsListView.OnScrollListener.SCROLL_STATE_FLING:// 滑动状态
threadFlag = false;
break;
case AbsListView.OnScrollListener.SCROLL_STATE_IDLE:// 停止
threadFlag = true;
startThread();
break;
case AbsListView.OnScrollListener.SCROLL_STATE_TOUCH_SCROLL:// 触摸listView
threadFlag = false;
break;
default:
// Toast.makeText(contextt, "default",
// Toast.LENGTH_SHORT).show();
break;
}
}
};
2、获取当前屏幕上显示的items:
mListView.getFirstVisiblePosition();
mListView.getLastVisiblePosition();

抱歉!评论已关闭.