【白话Android】为什么ListView能够实现成百上千条数据都不会OOM?

由于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,如果当前遗弃列表为空,那么就重新加载;如果不为空,则直接从遗弃列表里面拿。

有一个比较直观的图如下:


RecycleBin.png

由于屏幕大小有限,因此一个屏幕下容纳的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 {
    private int mResourceId;

    public PersonInfoAdapter(@NonNull Context context, int resource, @NonNull List 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;
    }
}

你可能感兴趣的:(【白话Android】为什么ListView能够实现成百上千条数据都不会OOM?)