由于AbsListView有个内部类RecycleBin,实现了RecycleBin机制,里面有维护两个列表,一个是正在使用的view,也就是在屏幕上能看见的view,一个是已经被遗弃的view,也就是屏幕上看不到的view。ListView有一个特点,就是所有子view的布局都是一样的,因此,就给了一个思路,就是不是每次都要生成一个view,而是对于需要生成view的时候,可以考虑复用看不见的view。
方案一:每次都重新生成view,由于ListView的子View的数量是由应用决定的,因此这个方案是不可控的,如果view的数量超级多,那么一不小心就会出现OOM。
方案二:目前Android采用的方案是使用RecycleBin机制。
(1)初始化的时候,只加载屏幕可见的几个View。
(2)在滑动的时候,将滑出去屏幕看不见的部分的View从可见View列表里面移除并放到遗弃列表中,同时对于新划入的View,需要先去遗弃View列表中获取View,如果当前遗弃列表为空,那么就重新加载;如果不为空,则直接从遗弃列表里面拿。
有一个比较直观的图如下:
由于屏幕大小有限,因此一个屏幕下容纳的View是有限的,因此,即使无数个数据,需要的View的空间也只有可见屏幕那个多个。
扩展
这个也是我们在自定义ListView适配器的时候应该进行优化的地方。可以参考源码ArrayAdapter.java的getView方法。在getView的时候,不能每次都重新加载一个新的View,而是先看下能不能复用原来的View。而getView的参数里面的convertView就是这样一个缓存。
Android中的源码如下:
@Override
public @NonNull View getView(int position, @Nullable View convertView,
@NonNull ViewGroup parent) {
return createViewFromResource(mInflater, position, convertView, parent, mResource);
}
private @NonNull View createViewFromResource(@NonNull LayoutInflater inflater, int position,
@Nullable View convertView, @NonNull ViewGroup parent, int resource) {
final View view;
final TextView text;
if (convertView == null) {
view = inflater.inflate(resource, parent, false);
} else {
view = convertView;
}
...
}
同时,在ListView优化方面,还有一个优化点,就是不要每次都去调用findViewById去获取控件的实例,而是可以借助自定义的ViewHolder将缓存存下来。
完整的示例代码如下:
public class PersonInfoAdapter extends ArrayAdapter<PersonInfo> {
private int mResourceId;
public PersonInfoAdapter(@NonNull Context context, int resource, @NonNull List<PersonInfo> objects) {
super(context, resource, objects);
mResourceId = resource;
}
@NonNull
@Override
public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
View view;
ViewHolder viewHolder;
if (convertView == null) {
view = LayoutInflater.from(getContext()).inflate(mResourceId, parent, false);
viewHolder = new ViewHolder();
viewHolder.imageView = (ImageView)view.findViewById(R.id.image);
viewHolder.textView = (TextView)view.findViewById(R.id.text);
view.setTag(viewHolder);
} else {
view = convertView;
viewHolder = (ViewHolder)view.getTag();
}
PersonInfo personInfo = getItem(position);
viewHolder.imageView.setImageResource(personInfo.getmImageId());
viewHolder.textView.setText(personInfo.getmName());
return view;
}
class ViewHolder {
ImageView imageView;
TextView textView;
}
}