搬书
- TaskApater中的getview方法,每次都将布局重新加载了一遍。当Listview快速滚动的时候,这就会成为性能的瓶颈。
- 在getview方法中还有一个convertView参数,这个参数用于将之前加载好的布局进行缓存,以便之后可以进行重用。
@NonNull
@Override
public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
Task task = getItem(position);
View view;
if (convertView == null){
view = LayoutInflater.from(getContext()).inflate(resourceId,parent,false);
} else {
view = convertView;
}
ImageView taskImage = (ImageView) view.findViewById(R.id.task_image);
TextView taskName = (TextView) view.findViewById(R.id.task_name);
taskImage.setImageResource(task.getImageId());
taskName.setText(task.getName());
return view;
}
- 可以看到。现在我们在getview方法中进行了判断,如果convertview为null,则使用LatoutInflater去加载布局,如果不为null则直接对convertview进行重用
还可以继续优化虽然现在已经不会再去重复加载布局,但是每次在getview方法中还是会调用view的findviewbyid的方法来获取一次控件的实例
- 我们可以借助一个VIewHolder来对这部分性能进行优化,修改TaskAdapter中的代码
public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
Task task = getItem(position);
View view;
ViewHolder viewHolder;
if (convertView == null){
view = LayoutInflater.from(getContext()).inflate(resourceId,parent,false);
viewHolder = new ViewHolder();
viewHolder.taskImage = (ImageView) view.findViewById(R.id.task_image);
viewHolder.taskName = (TextView) view.findViewById(R.id.task_name);
view.setTag(viewHolder);
} else {
view = convertView;
viewHolder = (ViewHolder) view.getTag();
}
viewHolder.taskName.setText(task.getName());
viewHolder.taskImage.setImageResource(task.getImageId());
return view;
}
class ViewHolder{
ImageView taskImage;
TextView taskName;
}
}
- 我们新建了一个内部类ViewHolder,用于对空间的实例进行缓存。当convertview中null时,创建一个viewholder对象,并且将控件的实例都放进viewHolder中。
当convertview不为null的时候,则调用View的getTag()方法,把ViewHolder重新取出。这样所有控件的实例都缓存在了viewHolder里,就没有必要 每次都通过findviewbyid的方法来获取控件实例了。