原文地址:Android ListView优化之getView频繁多次调用与ViewHolder工作原理
Android中我们经常会用到ListView,然后ListView到底是如何通过ViewHolder去优化的?
常见的适配器中利用ViewHolder去优化ListView的代码
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder viewHolder;
if (convertView == null) {
convertView = LayoutInflater.from(context).inflate(R.layout.item, null);
viewHolder = new ViewHolder();
viewHolder.textView = (TextView) convertView.findViewById(R.id.textView);
convertView.setTag(viewHolder);
} else {
viewHolder = (ViewHolder) convertView.getTag();
}
String name = datas.get(position);
viewHolder.textView.setText(name);
return convertView;
}
private class ViewHolder {
public TextView textView;
}
其实大家应该去思考一个问题,为什么要这么去写?大多数人可能只知道是优化的,但是是如何进行优化的呢?
关于getView的调用次数
举个例子:比如有5组数据要填充到listView。listView会先调用onMeasure,此时会调用5次getView。然后才调用onLayout,此时又会调用5次getView,这样就重复了。所以导致多次调用getView方法。
当ListView高度为wrap_content时:
<ListView
android:id="@+id/listview"
android:divider="#666666"
android:dividerHeight="1px"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
如下图:
这里牵扯一个问题,那就是布局优化的问题,如果我们将ListView的在这里插入代码片高度设置为wrap_content时ListView会去调用getView去动态计算高度,
这样的话导致ListView会再次调用getView去渲染视图。所以建议,将ListView的高度设置成一个固定的值或者match_parent,这样的话,会啊减少调用次数。
- 将ListView高度设置为match_parent时如下图:
<ListView
android:id="@+id/listview"
android:divider="#666666"
android:dividerHeight="1px"
android:layout_width="match_parent"
android:layout_height="match_parent" />
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="48dp"
android:orientation="vertical">
<TextView
android:id="@+id/textView"
android:layout_width="match_parent"
android:layout_height="48dp"
android:gravity="center"
android:text="万能适配器测试"
android:textColor="#000000"
android:textSize="18sp" />
</LinearLayout>
2.getView到底加载了多少个item布局与创建了多少个ViewHolder?findViewById多少次?
通过日志,我们发现,其实listView不是根据你设置的多少个item数据去加载布局,而是根据你手机的屏幕一屏能够
展示多少个item的数据(包括没有完全显示的item),也就是说
if (convertView == null) {
convertView = LayoutInflater.from(context).inflate(R.layout.item, null);
viewHolder = new ViewHolder();
viewHolder.textView = (TextView) convertView.findViewById(R.id.textView);
convertView.setTag(viewHolder);
}
这段加载布局的代码只走了你手机一屏能显示的个数,也就是说创建了这么多的Viewholder,但你滑动到别的也没的时候会复用第一屏的item然后我们去改变item中的值。由此可以看出,ListView不管记载多少数据都不会OOM,因为ListView始终只加载了一屏的数据这正是ListView的强大之处。其复用原理如图: