Listview是andorid中最常用的控件之一,但要用好这个控件并不是那么容易。不注意优化的使用经常出现页面卡顿,OOM等问题的出现。在此本人将自己的拙见整理汇总,归纳listview的优化措施。
1.复用convertView
2.viewHolder保存控件
3.分页加载
4.UI卡顿优化
5.OOM

复用convertView

这一条和下面一条是最常见的优化,相信你也会在各种网络资料或者参考书中见到此类优化。如果还不是很理解为何要这样处理的话,我举一个例子来类比一下这个过程。
相信大家都做过手扶电梯,台阶是由一块块板组成的,运行的过程中,人只需要站在一块板上,就会自动把人送到上方,然后人离开,该块板会在电梯内部重新运行到下方,再次携带一个人上来。
类比到listview,“台阶板”可以看做是子条目item的view,而“人”就是子条目item要显示的数据,屏幕中可见的item就相当于在电梯上的“人”和“板”,当listview往下滑直到某一个item不可见时,就相当于电梯的某块“板”把“人”送到上方直到离开电梯。在不进行优化的情况下,我们在getView方法中每次都会给子条目new一个view出来,当该view不可见后销毁它,把这个过程放到手扶电梯中,就是不断地在下方造“板”,把“人”送到上方后,销毁这块“板”。看到这里相信你肯定在想,这得多浪费资源啊!所以电梯采用的是将“板”在电梯内部重新运行到下方,这样同一块“板”就可以不断地运送“人”。
google工程师在设计listview的时候已经考虑到这种情况,因此在getView的参数中有一个convertView,它是用来保存滑出屏幕范围的item的view,即运送完“人”转入电梯内部的“板”。我们要做的就是获取到这个convertView,重新赋予数据,作为新item来展示,即这块“板”运行到下方,再次运送一个“人”到上方的过程。
电梯将“板”运送到下方需要时间,因此通常电梯“板”的数量是你能看到的2倍。而在内存中一切都是非常快速的,当一个item的view不可见,被保存到convertView后,马上可以作为新的数据的载体,用作下一个item的view来使用,因此,listview仅需要一个屏幕可见item的个数 + 1(convertView)就可以显示所有的数据,无需创建更多的view再销毁,从而达到优化目的。

@Override
    public View getView(int position, View convertView, ViewGroup parent) {
        if (convertView == null) {
            convertView = View.inflate(mContext, android.R.layout.simple_list_item_1, null);
                Log.i("test", "inflate a View");
        }
        TextView textView = (TextView) convertView.findViewById(android.R.id.text1);
        textView.setText(data.get(position));
        return convertView;
    }

通过复用后,测试手机的屏幕可以看到13个item,无论我如何滑动,也只会创建14次view。

Android ListView优化 listview优化方案_分页

viewHolder保存控件

通过上面的代码可以看到,view是不再过多的创建了,但是findViewById这个方法还是在每次getView的时候都会执行,该方法需要遍历整个布局文件来找到对应的ID,是很耗费资源的,有没有什么办法减少这个过程那?
想象一下让你在一本书中找到某一句话,每次你都要从头翻到尾,直到找到那句话,这是非常耗时的。而如果你在第一次找到后,把对应的页数记录下来,下次再让你找的时候,直接翻到对应页面就可以快速地找到那句话。在listview中我们也可以采用同样的方法,将找到的控件记录在内存中,这样下次再找控件的时候直接从内存中取出即可。
通常的做法是定义一个静态内部类ViewHolder,里面仅有成员变量来保存控件:

@Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;
        if (convertView == null) {
            holder = new ViewHolder();
            convertView = View.inflate(mContext, android.R.layout.simple_list_item_1, null);
            holder.textView = (TextView) convertView.findViewById(android.R.id.text1);
            convertView.setTag(holder);
            Log.i("test", "inflate a View");
        }else {
            holder = (ViewHolder) convertView.getTag();
        }
        holder.textView.setText(data.get(position));
        return convertView;
        }
    }   
private static class ViewHolder{
    TextView textView;
}

是不是会有疑问:为什么要定义成static的那?为什么要叫viewHolder?
非静态内部类拥有外部类的强引用,因此为了避免对外部类(可能是 Activity)对象的引用造成内存泄露,那么最好将内部类声明为 static 的;
其实ViewHolder这个名字可以随意的取,但你写的代码不是只有你自己看,如果你定义为“abc”,那只有你知道这是干嘛的,而其他人不知道,而定义成ViewHolder可以达到见名知意。
分页加载
当listview需要显示的数据较多,数据加载的时间过长,避免用户过长时间的等待数据刷新,可以采用的分页加载的方式对listview进行优化。分页加载的思路就是将大量地数据分成多个小份,每次只加载其中的一份数据,加载新数据的时机即用户滑动到listview底部时。为提升用户体验,在加载数据时在listview底部显示一个progressBar来告诉用户正在加载新数据,并在完成加载后取消这个显示。下面是实现的步骤:
1.inflate一个listview的footer,用于显示progressBar

footer = View.inflate(this, R.layout.footer, null);

2.之前在网上看到的资料都会在setAdapter之前添加footer,setAdater之后移除

//listView.addFooterView(footer);
    adapter = new MyAdapter(this, data);
    listView.setAdapter(adapter);
    //listView.removeFooterView(footer);

此处肯定有疑惑,为啥要如此操作?先来看下setAdapter源码中的一段:

if (mHeaderViewInfos.size() > 0|| mFooterViewInfos.size() > 0) {
    mAdapter = new HeaderViewListAdapter(mHeaderViewInfos, mFooterViewInfos, adapter);
} else {
    mAdapter = adapter;
}

mFooterViewInfos这个变量一看便知是footer的信息类,当有footer时,传入的adapter会进行包装,因此当时看到的资料才会有如此操作,目的是让adater进行包装。但实际上是多此一举的。继续看下addFooterView方法的源码:

public void addFooterView(View v, Object data, boolean isSelectable) {
   final FixedViewInfo info = new FixedViewInfo();
   info.view = v;
   info.data = data;
   info.isSelectable = isSelectable;
   mFooterViewInfos.add(info);
   mAreAllItemsSelectable &= isSelectable;

   // Wrap the adapter if it wasn't already wrapped.
   if (mAdapter != null) {
      if (!(mAdapter instanceof HeaderViewListAdapter)) {
        mAdapter = new HeaderViewListAdapter(mHeaderViewInfos, mFooterViewInfos, mAdapter);
      }

       // In the case of re-adding a footer view, or adding one later on,
       // we need to notify the observer.
       if (mDataSetObserver != null) {
           mDataSetObserver.onChanged();
       }
    }
}

可以看到内部有判断,如果已经setAdpter,即mAdapter != null条件满足,就会将该adapter进行包装。因此无需做上述操作,仅在需要的时候直接setFooterView就可以了。
3.设置滑动监听,用于判断用户滑动到底部:

listView.setOnScrollListener(new OnScrollListener() {           
    @Override
    public void onScrollStateChanged(AbsListView view, int scrollState) {               
        if(scrollState==SCROLL_STATE_IDLE){
            int lastVisiblePosition = listView.getLastVisiblePosition();
            if(lastVisiblePosition==adapter.getCount()-1){
//避免重复加载数据
                if(isBottom){
                    return;
                }
                isBottom=true;
                listView.addFooterView(footer);
                listView.setSelection(lastVisiblePosition);
                new Thread(new MyTask()).start();
            }
        }           
    }           
    @Override
    public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {}
});

当滑动到底部时添加footer显示progressBar,setSelection调整listview的显示位置,同时开启子线程加载数据。
4.加载数据:

private static class MyTask implements Runnable{
        @Override
        public void run() { 
            List<String> newData = new ArrayList<String>();
            //模拟数据
            count+=20;
            //模拟无更多数据加载情况
            if(count>150){
                handler.sendEmptyMessage(NO_MORE_DATA);
            }else{
                for (int i = count-20; i < count; i++) {                    
                    newData.add("item" + i);
                }
                //模拟耗时操作    
                SystemClock.sleep(2000);
                //通知主线程更新UI
                handler.obtainMessage(RESULT_OK, newData).sendToTarget();   
            }
        }   
    }

5.更新UI

private static Handler handler = new Handler(){
        public void handleMessage(android.os.Message msg) {
            switch (msg.what) {
            case RESULT_OK:
                List<String> newData = (List<String>) msg.obj;
                data.addAll(newData);
                //移除footer
                listView.removeFooterView(footer);
                //更新数据显示
                adapter.notifyDataSetChanged();
                isBottom = false;
                break;
            case NO_MORE_DATA:
                View pb = footer.findViewById(R.id.footer_pb);
                pb.setVisibility(View.GONE);
                View tView = footer.findViewById(R.id.footer_tv);
                tView.setVisibility(View.VISIBLE);  
                isBottom = false;
                break;
            default:
                break;
            }
        }

NO_MORE_DATA的情况是模拟数据加载完成,更换footer显示内容“无更多数据加载。。。”

UI卡顿优化

1.避免耗时操作在主线程完成,运行所需时间超过16毫秒的任务都会引起页面的卡顿,因此超过16毫秒的任务都应放在子线程完成。
2.UI视图避免过于复杂,过多的层级嵌套,重复的布局可以使用include标签代替,不常用的视图(例如点击才显示)可以使用viewStub标签。可以使用SDK自带的hierarchyviewer工具(androidSDK\tools目录下)查看视图层级。

OOM

如果在listview中显示了大量图片,不进行处理的话,很容易出现OOM的情况。常见的处理方式有:
1.加载图片时对分辨率进行缩放,在android手机中一个像素点会占用4byte的内存,分辨路过大的图片会占用更多的内存。
2.使用弱引用来保存bitmap,并及时调用recycle()方法释放内存。
3.使用已有框架,如ImageLoader、Glide 、Picasso、Fresco 等,这些框架都做了相应处理。