Android中我们经常会用到ListView,然后ListView到底是如何通过ViewHolder去优化的?
1.常见的适配器中利用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; }
其实大家应该去思考一个问题,为什么要这么去写?大多数人可能只知道是优化的,但是是如何进行优化的呢?
2.关于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的强大之处。其复用原理如图: