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对象,做耗时操作
