Android性能优化-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 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>的方法缓存,不过不好用了,现在大多都用
LruCache,

public class MemoryCache {
    private LruCache mLruCache = null;
    public MemoryCache(){
        long maxMemory  = Runtime.getRuntime().maxMemory();//最大内存 默认是16M
        mLruCache = new LruCache((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对象,做耗时操作

你可能感兴趣的:(Android性能优化-ListView优化)