美文网首页Android开发Android技术知识Android进阶之路
深入解析 Android 中 View 的工作原理(上)

深入解析 Android 中 View 的工作原理(上)

作者: cff70524f5cf | 来源:发表于2019-02-20 22:29 被阅读1次

Android中的任何一个布局、任何一个控件其实都是直接或间接继承自View实现的,当然也包括我们在平时开发中所写的各种炫酷的自定义控件了,所以学习View的工作原理对于我们来说显得格外重要,本篇文章,我们将一起深入学习Android中View的工作原理。

ViewRoot和DecorView

1、ViewRoot对应于ViewRootImpl类,是连接WindowManager和DecorView的纽带,View的三大流程均是通过ViewRoot来完成的。在ActivityThread中,当Activity对象被创建完毕后,会将DecorView添加到Window中,同时会创建ViewRootImpl对象,并将ViewRootImpl对象和DecorView建立关联。

2、View的绘制流程从ViewRoot的performTraversals开始,经过measure、layout和draw三个过程才可以把一个View绘制出来,其中measure用来测量View的宽高,layout用来确定View在父容器中的放置位置,而draw则负责将View绘制到屏幕上。

3、performTraversals会依次调用performMeasure、performLayout和performDraw三个方法,这三个方法分别完成顶级View的measure、layout和draw这三大流程。其中performMeasure中会调用measure方法,在measure方法中又会调用onMeasure方法,在onMeasure方法中则会对所有子元素进行measure过程,这样就完成了一次measure过程;子元素会重复父容器的measure过程,如此反复完成了整个View数的遍历。

measure过程决定了View的宽/高,完成后可通过getMeasuredWidth/getMeasureHeight方法来获取View测量后的宽/高。Layout过程决定了View的四个顶点的坐标和实际View的宽高,完成后可通过getTop、getBotton、getLeft和getRight拿到View的四个定点坐标。Draw过程决定了View的显示,完成后View的内容才能呈现到屏幕上。

DecorView作为顶级View,一般情况下它内部包含了一个竖直方向的LinearLayout,里面分为两个部分(具体情况和Android版本和主题有关),上面是标题栏,下面是内容栏。在Activity通过setContextView所设置的布局文件其实就是被加载到内容栏之中的。

MeasureSpec

1、MeasureSpec很大程度上决定一个View的尺寸规格,测量过程中,系统会将View的layoutParams根据父容器所施加的规则转换成对应的MeasureSpec,再根据这个measureSpec来测量出View的宽/高。

2、MeasureSpec代表一个32位的int值,高2位为SpecMode,低30位为SpecSize,SpecMode是指测量模式,SpecSize是指在某种测量模式下的规格大小。

MpecMode有三类:

1、UNSPECIFIED 父容器不对View进行任何限制,要多大给多大,一般用于系统内部

2、EXACTLY 父容器检测到View所需要的精确大小,这时候View的最终大小就是SpecSize所指定的值,对应LayoutParams中的match_parent和具体数值这两种模式。

3、AT_MOST 父容器指定了一个可用大小即SpecSize,View的大小不能大于这个值,不同View实现不同,对应LayoutParams中的wrap_content。

当View采用固定宽/高的时候,不管父容器的MeasureSpec的是什么,View的MeasureSpec都是精确模式兵其大小遵循Layoutparams的大小。 当View的宽/高是match_parent时,如果他的父容器的模式是精确模式,那View也是精确模式并且大小是父容器的剩余空间;如果父容器是最大模式,那么View也是最大模式并且起大小不会超过父容器的剩余空间。 当View的宽/高是wrap_content时,不管父容器的模式是精确还是最大化,View的模式总是最大化并且不能超过父容器的剩余空间。

对于DecorView,它的MeasureSpec由Window的尺寸和其自身的LayoutParams来共同确定,对于普通的View,其MeasureSpec由父容器的MeasureSpec和自身的Layoutparams来共同确定。

对于 DecorView,在ViewRootImpl源码中的measureHierarchy有如下一段代码:

我们查看一下getRootMeasureSpec方法的源码:

    从上面的代码中就可以很容理解DecorView的MeasureSpec是如何产生的,rootDimension就是DecorView自身的LayoutParams,然后会根据这个值进行判断LayoutParams.MATCH_PARENT:DecorView的MeasureSpec被赋值为精确模式,DecorView的大小就是Window的大小。

ViewGroup.LayoutParams.WRAP_CONTENT:DecorView的MeasureSpec被赋值为最大模式,DecorView的大小不定,但是不能超过Window的大小

默认情况:DecorView的MeasureSpec被赋值为精确模式,DecorView的大小为自身LayoutParams设置的值,也就是rootDimension

接着是对于普通的View,也就是布局中的View,它的Measure过程由ViewGroup传递而来,其中有一个方法是measureChildWithMargins:

在对子view进行measure之前会先调用getChildMeasureSpec方法来获取子view的MeasureSpec,从这段代码就可以看出来子view的MeasureSpec的确定与父容器的MeasureSpec(parentWidthMeasureSpec)还有自身的LayoutParams(lp.height和lp.width),还有View自己的Margin和Padding有关。

接下来查看getChildMeasureSpec方法源码:

这里参数中的padding是指父容器的padding,这里是父容器所占用的空间,所以子view能使用的空间要减去这个padding的值。同时这个方法内部其实就是根据父容器的MeasureSpec结合子view的LayoutParams来确定子view的MeasureSpec。

由于篇幅限制,这一篇就和大家介绍到这里,想进一步浏览,请关注下一篇《深入解析 Android 中 View 的工作原理(下)》。

最后附上小编整理出来的Android相关的学习思维导图,让大家有个学习的方向。

Android进阶

Android前沿技术

Flutter

移动架构师

需要这些资料的大伙关注+点赞+加群:185873940 免费获取!

群内还有许多免费的关于高阶安卓学习资料,包括高级UI、性能优化、架构师课程、 NDK、混合式开发:ReactNative+Weex等多个Android技术知识的架构视频资料,还有职业生涯规划及面试指导。

相关文章

网友评论

    本文标题:深入解析 Android 中 View 的工作原理(上)

    本文链接:https://www.haomeiwen.com/subject/xtnkyqtx.html