ListView优化

Android性能优化-ListView
ListView的优化主要分为以下几点

1 convertView的复用
ListView每次滚动都会调用getView方法,所以优化getVieiw是重中之重

convertView介绍
convertView是刚刚滚动出可见区域的View的引用,此时它已经不可见,所以应该被复用以减少View的创建

优化代码
View view = null;//getView方法要返回的View
if(convertView == null){//如果当前没有可以复用的View
view = LayoutInflater.from(context).inflate(resourceId,null);//那么就从XML文件生成一个View
}else{//否则
view = convertView;//就使用可以复用的View
}
优化原因
LayoutInflater.inflate(resourceId,View)这个方法是用来通过pull的解析方式从XML文件生成一个View对象的,如果有成千上万
个Viwe都要去解析XML生成View,会非常消耗性能

2ViewHolder的使用
优化代码
ViewHolder viewHolder = null;
View view = null;//getView方法要返回的View
if(convertView == null){//如果当前没有可以复用的View
viewHolder = new ViewHolder();
view = LayoutInflater.from(context).inflate(resourceId,null);//那么就从XML文件生成一个View
viewHolder.resourceViewName = view.findViewById(resouceViewId);//从XML中找到对应的View
view.setTag(viewHolder);//将ViewHolder设置在当前ItemView的tag里面
}else{//否则
view = convertView;//就使用可以复用的View
viewHolder = (ViewHolder)convertView.getTag();//从复用的View中取出viewHoder
}
viewHolder

class ViewHolder {
TextView name;
}
优化原因
findViewById这个方法是从ViewGroup的子View里面循环遍历找id与给出的ID相同的子View,还是比较耗时的,

/*ViewGroup的FindViewByID源码/

/**
 * {@hide}
 */
@Override
protected <T extends View> T findViewTraversal(@IdRes int id) {
    if (id == mID) {
        return (T) this;
    }

    final View[] where = mChildren;
    final int len = mChildrenCount;

    for (int i = 0; i < len; i++) {
        View v = where[i];

        if ((v.mPrivateFlags & PFLAG_IS_ROOT_NAMESPACE) == 0) {
            v = v.findViewById(id);

            if (v != null) {
                return (T) v;
            }
        }
    }

    return null;
} 

3图片"三级缓存"加载优化
阐述
图片加载顺序,应该为,内存–本地–网络

1、内存缓存 优先加载,速度最快
2、本地缓存 次优先加载 速度稍快
3、网络缓存 最后加载 速度由网络速度决定(浪费流量)
代码(缓存到本地,从网络获取就不写了)
主要写一下缓存到内存中的方法,
据说以前使用HashMap<String,SoftReference>的方法缓存,不过不好用了,现在大多都用
LruCache,

public class MemoryCache {
private LruCache<String,Bitmap> mLruCache = null;
public MemoryCache(){
long maxMemory = Runtime.getRuntime().maxMemory();//最大内存 默认是16M
mLruCache = new LruCache<String,Bitmap>((int)(maxMemory/8)){
@Override
protected int sizeOf(String key, Bitmap value) {
//int byteCount = value.getByteCount();
//得到图片的字节数
int byteCount = value.getRowBytes() * value.getWidth();
return byteCount;
}
};
}
//从内存获取
public Bitmap getFromMemory(String url){
return mLruCache.get(url);
}
//缓存到内存
public void setToMemory(String url,Bitmap bitmap){
mLruCache.put(url,bitmap);
}
}
优化原因
从网络加载图片或者本地加载图片都比较耗时,加上Android16ms的刷新UI频率,会造成卡顿
从内存获取速度相对较快,以上只是放入内存的方法,当然压缩什么的就没有写,只是简单介绍存入内存的原理

4图片加载再次优化
导论
很多情况下ListView需要加载显示网络图片,我们尽量不要在ListView滑动的时候加载网络图片,
那样会使ListView变得卡顿所以我们要监听ListView的状态,如果ListView滑动(SCROLL_STATE_TOUCH_SCROLL)
或者猛滑(SCROLL_STATE_FLING)的时候,停止加载图片,否则加载图片

优化代码
listView.setOnScrollListener(new AbsListView.OnScrollListener() {
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
if (scrollState == OnScrollListener.SCROLL_STATE_IDLE) {//list停止滚动时加载图片
loadImage(startPos, endPos);// 异步加载图片 ,只加载可以看到的图片
}
}

@Override
public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
//设置当前屏幕显示的起始pos和结束pos
startPos = firstVisibleItem;
endPos = firstVisibleItem + visibleItemCount;
if (endPos >= totalItemCount) {
endPos = totalItemCount - 1;
}
}
});
优化原因
从用户的角度讲,快速滑动的时候,用户不需要看到当前内容

5 onClickListener处理
导论
有时候出了onItemClickListener之外我们还会用到Item上其他位置的点击事件
一般情况下我们是在getView方法中,一个一个设置,就像

holder.img.setOnClickListener(new onClickListener());
这样每个都设置了一个新的OnClickListener对象,不太好

优化方案
直接在ViewHolder中设置一个position,然后viewHolder implements OnClickListener

class ViewHolder implements OnClickListener{
int position;
TextView name;

public void setPosition(int position){  
    this.position = position;  
}  

@Override  
public void onClick(View v) {  
    switch (v.getId()){  
        //XXXX  
    }  
}  

}
然后再getView中设置的时候设置自己就行了

ViewHolder viewHolder = null;
View view = null;//getView方法要返回的View
if(convertView == null){//如果当前没有可以复用的View
viewHolder = new ViewHolder();
view = LayoutInflater.from(context).inflate(resourceId,null);//那么就从XML文件生成一个View
viewHolder.resourceViewName = view.findViewById(resouceViewId);//从XML中找到对应的View
viewHolder.setPosition(position);//设置位置
viewHolder.name.setOnClickListener(viewHolder);//设置ClickListener
view.setTag(viewHolder);//将ViewHolder设置在当前ItemView的tag里面
}else{//否则
view = convertView;//就使用可以复用的View
viewHolder = (ViewHolder)convertView.getTag();//从复用的View中取出viewHoder
}

6 总结
总之,宗旨就是少在getView里面new对象,做耗时操作

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值