RecyclerView源码分析(二)--测量流程

阅读本文您大概需要4.33分钟。

相关系列文章

RecyclerView源码分析(一)–整体设计

在上一篇文章中主要讲解了RecyclerView内部的大体设计结构。因为是从结构出发,所以多少对细节讲的云里雾里。那么从这一篇开始要落实到代码细节中了。

既然RecyclerView是一个GroupView,那么也就是一个View。那么View的绘制过程是measure到layout到draw的一个顺序。然而一个GroupView的目的是盛放其它View的,那么最主要的还是其measure和layout过程。那么我们今天就来看看RecyclerView的measure过程。

PS:源码版本为24.1.1,如果下面与你的源码有出入,请核实版本是否相同。

RecyclerView的Measure过程

如果你这个时候也打开了源码,你应该会发现RecyclerView的onMeasure方法很长。那么我们先将其分情况讨论。

  1. 没有LayoutManager的情况
  2. 有LayoutManager并开启了自动测量
  3. 有LayoutManager但没有开启自动测量
@Override
protected void onMeasure(int widthSpec, int heightSpec) {
    if (mLayout == null) {
        // 1. 没有LayoutManager的情况
        defaultOnMeasure(widthSpec, heightSpec);
        return;
    }
    if (mLayout.mAutoMeasure) {
        // 2. 有LayoutManager并开启了自动测量
        ……
    } else {
        // 3. 有LayoutManager但没有开启自动测量
        ……
    }
}

第一种情况十分简单,就是执行了defaultOnMeasure方法。里面就是计算并设置了RecyclerView的长宽值。下来我们介绍另外两种情况。

RecyclerView的自动测量过程

在RecyclerView的早期版本,当你为其设置了wrap_content值,但在其中的内容改变的时候,RecyclerView并不能改变其大小来适应内部的内容。因此后来加入了自动测量机制,来解决这个问题。而且现在我们常用的三个LayoutManager,在其构造函数中,均已经设置了开启自动测量。所以我们现在可以放心大胆的为RecyclerView设置wrap_content。

那么我们来看一下RecyclerView的自动测绘过程:

protected void onMeasure(int widthSpec, int heightSpec) {
    ……
    if (mLayout.mAutoMeasure) {
        // 第一部分:
        // 1) 首先执行LayoutManager的onMeasure方法。
        // 2) 检查如果width和height都已经是精确值,那么就不用再根据内容进行计算所
        // 需要的width和height,那么跳过之后的步骤。如果有其中任何一个值不是精确值,
        // 则进入到下面计算所需长宽的步骤。
        final int widthMode = MeasureSpec.getMode(widthSpec);
        final int heightMode = MeasureSpec.getMode(heightSpec);
        final boolean skipMeasure = widthMode == MeasureSpec.EXACTLY
                && heightMode == MeasureSpec.EXACTLY;
        mLayout.onMeasure(mRecycler, mState, widthSpec, heightSpec);
        if (skipMeasure || mAdapter == null) {
            return;
        }
        // 第二部分:
        // 1) 开启布局流程计算出所有Child的边界
        // 2) 然后根据计算出的Child的边界计算出RecyclerView的所需width和height
        // 3) 检查是否需要再次测量
        if (mState.mLayoutStep == State.STEP_START) {
            dispatchLayoutStep1();
        }
        mLayout.setMeasureSpecs(widthSpec, heightSpec);
        mState.mIsMeasuring = true;
        dispatchLayoutStep2();

        // 布局过程结束,根据Children中的边界信息计算并设置RecyclerView长宽的测量值
        mLayout.setMeasuredDimensionFromChildren(widthSpec, heightSpec);
        
        // 检查是否需要再此测量。如果RecyclerView仍然有非精确的宽和高,或者这里还有至
        // 少一个Child还有非精确的宽和高,我们就需要在此测量。
        if (mLayout.shouldMeasureTwice()) {
            mLayout.setMeasureSpecs(
                    MeasureSpec.makeMeasureSpec(getMeasuredWidth(), MeasureSpec.EXACTLY),
                    MeasureSpec.makeMeasureSpec(getMeasuredHeight(), MeasureSpec.EXACTLY));
            mState.mIsMeasuring = true;
            dispatchLayoutStep2();
            mLayout.setMeasuredDimensionFromChildren(widthSpec, heightSpec);
        }
    } else {
        ……
    }
}

自动测绘过程可以分为两部分:

第一部分:

  1. 首先执行LayoutManager的onMeasure方法。
  2. 检查如果width和height都已经是精确值,那么就不用再根据内容进行计算所需要的width和height,那么跳过之后的步骤。如果有其中任何一个值不是精确值,则进入到下面计算所需长宽的步骤。

第二部分:

  1. 开启布局流程,计算出所有Child的边界。
  2. 然后根据计算出的Child的边界计算出RecyclerView的所需width和height,并设置。
  3. 检查是否需要再次测量,如果需要则在此进行测量。

注意:
因为自动绘制过程中执行了布局流程,那么在之后布局的时候会检查是否已经进行过布局流程,如果已经进行过布局流程,则会跳过进行过的布局流程,不会造成重复操作。(布局流程有三步)

RecyclerView的非自动测绘流程

当我们使用系统提供的那三个LayoutManager的时候,默认是开启自动测绘的。除非,你在初始化LayoutManager之后,自己通过setAutoMeasureEnabled(false)方法设置成false。不然不会走到非自动测绘流程的。但是如果我们是使用的自己自定义的LayoutManager,而且我们自定义的LayoutManager又没有在初始化的时候开启自动测绘,那么默认将会是不开启自动测绘。这个时候会走到非自动测绘流程。

那么接下来看一看非自动测绘流程。

protected void onMeasure(int widthSpec, int heightSpec) {
    ……
    if (mLayout.mAutoMeasure) {
        ……
    } else {
        // 第一部分:如果RecyclerView已经设置了Size固定,则执行LayoutManager的onMeasure方法
        if (mHasFixedSize) {
            mLayout.onMeasure(mRecycler, mState, widthSpec, heightSpec);
            return;
        }
        // 第二部分:
        // 1) 如果在测量的过程中有数据有更新,则先处理更新的数据
        // 2) 执行自定义测量流程,这需要自定义的LayoutManager实现onMeasure方法。
        if (mAdapterUpdateDuringMeasure) {
            eatRequestLayout();
            processAdapterUpdatesAndSetAnimationFlags();

            if (mState.mRunPredictiveAnimations) {
                mState.mInPreLayout = true;
            } else {
                // consume remaining updates to provide a consistent state with the layout pass.
                mAdapterHelper.consumeUpdatesInOnePass();
                mState.mInPreLayout = false;
            }
            mAdapterUpdateDuringMeasure = false;
            resumeRequestLayout(false);
        }
        // 处理完新更新的数据,然后执行自定义测量操作。
        if (mAdapter != null) {
            mState.mItemCount = mAdapter.getItemCount();
        } else {
            mState.mItemCount = 0;
        }
        eatRequestLayout();
        mLayout.onMeasure(mRecycler, mState, widthSpec, heightSpec);
        resumeRequestLayout(false);
        mState.mInPreLayout = false; // clear
    }
}

RecyclerView的非自动化测绘流程也可以分为两部分:

第一部分:
在固定尺寸模式下,直接执行LayoutManager的onMeasure方法。

第二部分:
不在固定尺寸模式下

  1. 如果在测量的过程中有数据有更新,则先处理更新的数据
  2. 执行自定义测量流程,这需要自定义的LayoutManager实现onMeasure方法。

总结

一般情况下,进入的都是自动测量模式。

最不常见的是没有设置LayoutManager的模式。

最后需要注意的是:我们自定义LayoutManager的时候要根据自己的需求,决定是否要开启自动测量。如果需要开启,则要主动调用setAutoMeasureEnabled(true),否则默认是不开启的。

感谢您的阅读,如果您觉得本文对你有所帮助,请不要吝啬您的喜欢,您的喜欢是我写作的动力。

    原文作者:柴泽建_Jack
    原文地址: https://www.jianshu.com/p/4b8d6e5004d5
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞