Android ListView 优化原理

先来看看我们之前的项目使用的Adapter的时候是如何开发的呢,先拿一个最简单的Adapter举个例子,这个列表每行只显示一行文字

public class StringAdapter extends BaseListAdapter<String> {

    public StringAdapter(Context context, List<String> list) {
        super(context, list);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {

        if (convertView == null) {
            convertView = LayoutInflater.from(mContext).inflate(R.layout.item_simple, null);
        }
        TextView tv_simple = convertView.findViewById(R.id.tv_simple);
        tv_simple.setText(mList.get(position));

        return convertView;
    }
}

大家可以看到

  View convertView = LayoutInflater.from(mContext).inflate(R.layout.item_simple, null);

我们使用这行代码创建了一个View 。就是我们显示的Item的View。

if (convertView == null) {
            convertView = LayoutInflater.from(mContext).inflate(R.layout.item_simple, null);
        }

我们的代码里是这样写的,做了一个判断,如果不加这个判断直接创建可以不可以呢?答案肯定是不可以啦。那么这行代码的作用是什么呢?

View的复用

举个例子大家就很容易理解了。
假设我的列表有100条数据。比如淘宝的商品列表 搜索之后会返回很多数据100都是少的,那么我们是不是创建100个ItemView每个ItemView 保存一条数据,也就是位每条数据创建一个View呢,也不是不行,知识100View其实已经非常费内存了。这样做之后,很容易出现两个问题,第一个就是卡顿,滑动页面时候容易卡顿,第二个就是直接程序内存溢出。如果1000条数据这样做,基本好手机 都得卡的不要不要的。
所以设计ListView的时候就考虑了这个问题,就是View的复用,虽然我列表有100条数据,但是用户看到的其实可能只有5条,那么我只需要创建5个View就够用户使用了,当用户滑动的时候,最上面的用户已经看不到了,那么我就可以直接拿过来放到最后面直接使用,每个View的布局是一样的,只是内容不一样,那么我只要 对这个View重新赋值就行了,而不是新建一个项目。这样就达到了View复用的目的。

我们之前那个判断就是为了实现这个功能而做的如果不加那个判断,每次滑动的时候都会创建新的View。所以一般情况写ListView都必须加那个判断。效率提升 最少一倍。

那么影响ListView的性能的是什么呢

最影响效率的就是创建View。我们通过上面的代码就可以解决了。

第二个呢就是findViewById()。这个方法的效率其实并不高。对于listview如果页面有非常多的控件,通过这个方法寻找控件,在有的手机都容易产生卡顿,所以我们希望就是每个ItemView我们在获取这个控件之后我们能缓存起来,这样在滑动的时候不需要findViewById()而是直接使用就好了,那么如何缓冲呢这里就有了ViewHolder这个东西。其实他只是一个普通的Java类。没有什么特别的,他里面知识放了Adapter的View的引用,因为每次都保存起来,以后使用就不需要findViewById()而已。
看看代码吧

  @Override
    public View getView(int position, View convertView, ViewGroup parent) {

        if (convertView == null) {
            convertView = LayoutInflater.from(mContext).inflate(R.layout.item_simple, null);
        }
        
        ViewHolder viewHolder = (ViewHolder) convertView.getTag();
        if(viewHolder == null){
            viewHolder = new ViewHolder();
            viewHolder.tv_simple = convertView.findViewById(R.id.tv_simple);
            convertView.setTag(viewHolder);
        } 

         viewHolder.tv_simple.setText(mList.get(position));

        return convertView;
    }
    class ViewHolder{
        TextView tv_simple; 
    }

代码只是改了需要的部分,比之前复杂了很多,但是带来的效率确实实实在在的,如果一个View 使用过那么他就会保持这个ViewHolder,然后我们可以方便的使用了,这样写之后,一般的页面运行就非常快了

ViewHolder的作用就是保存Ttem内的控件的引用,避免每次都需要findView。

知道ViewHolder的作用之后就好理解RecyclerView 里面为什么默认就要求使用ViewHolder了吧,因为效率高嘛。所以ViewHolde变成了一个强制使用的类

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容