理解Android中的MeasureSpec

article/2025/8/25 16:58:55

文章收藏的好句子:永远要相信美好的事情即将发生。

ps:本文源码是基于 Android Api 31 来分析的

目录

1、MeasureSpec

     1、1 SpecMode

     1、2 MeasureSpec 的 int 值和 LayoutParams 的对应关系

1、MeasureSpec

我们在 Android 手机上看到的界面,其实就是 View,显示 View 的过程中其实是要经过 View 的测量的,View 的测量就需要用到 MeasureSpec,那么这个 MeasureSpec 是什么呢?MeasureSpec 翻译成中文就是 “测量规格” 或者 “测量说明书” ,MeasureSpec 在一定程度上决定了一个 View 的尺寸规格,为什么说是一定程度上?是因为这个过程还受父容器的影响,父容器影响View 的MeasureSpec的创建过程;在测量过程中,系统会将View的LayoutParams 根据父容器所加限制的规则转换成对应的 MeasureSpec,最后根据这个 MeasureSpec 来测量出 View 的测量宽高。

这里的 MeasureSpec 有2层意思,一种是 MeasureSpec 对象,一种是 MeasureSpec 的 int 值;我们说一下 MeasureSpec 的 int 值,MeasureSpec 的 int 值是32位的对不对?那好,高2位就是 SpecMode,低30位就是 SpecSize;SpecMode 和 SpecSize 又是什么东东呢?SpecMode 是测量模式,SpecSize 是某种测量模式下的测量大小。

为了更好的理解 MeasureSpec 的 int 值,我们先看一下 View 的静态内部类 MeasureSpec ;

ea8eee947d384d720f4357e7755a5ecf.png

看注释1,makeMeasureSpec 方法中的参数 size 和 mode 其实就是 SpecSize 和 SpecMode,MeasureSpec 对象通过 makeMeasureSpec 方法将 SpecSize 和 SpecMode 打包成一个 MeasureSpec 的 int 值来避免过多的对象内存分配;看注释2和注释3,MeasureSpec 对象可以通过解包的形式得出它原始的 SpecMode 和 SpecSize 。

1、1 SpecMode

SpecMode 有三类,每一类都有它的含义,我们先看一下 View 的内部类 MeasureSpec 对 SpecMode 声明的是哪三类;

64640a79464c6eaec200b8e2ec448a1e.png

看到注释4、5、6 没有,SpecMode 的三类分别是 UNSPECIFIED、EXACTLY 和 AT_MOST ,下面对它们的含义进行说明一下;

UNSPECIFIED :父容器不对子 View 做任何的限制,想要多大就有多大,一般用于系统内部使用。

EXACTLY : 父容器已经测量出子 View 需要的精确大小,子 View 的最终大小其实就是 SpecSize 所指定的值,它对应于 match_parent 和具体的数值这两种模式;具体的数值应该理解吧?比如说200dp的数值。

AT_MOST : 父容器指定了一个可用大小 SpecSize,子 View 的大小不能超过这个值(SpecSize),具体是什么值要看子 View的子 View 的大小,它对应于 wrap_content 这个值。

1、2 MeasureSpec 的 int 值和 LayoutParams 的对应关系

这里对 MeasureSpec 的转换有2种,一种是顶级的 View(DecorView),一种是普通的 View,我们先看顶级 View 的 MeasureSpec 的转换是怎么样的;先看2段代码,看一下 ViewRootImpl 的 measureHierarchy 方法和 getRootMeasureSpec 方法;

453ec326d7d9ddfaf3aeebe72b38bb13.png

f0c08d78ae566b3f8fbb8cf0de6d65dd.png

看注释7和注释8,参数 desiredWindowWidth 和 desiredWind-owHeighgt 就是屏幕尺寸大小,参数 lp.width 和 lp.height 就是子 View 的 LayoutParams,所以顶级的 View(DecorView)的 MeasureSpec 是由窗口的大小和自身的 View(DecorView) 来决定的。

看注释9、10、11,DecorView 的 MeasureSpec 就由ViewGroup.Layout-Params.MATCH_PARENT 、ViewGroup.LayoutParams.WRAP_CONTENT和固定大小(注释11的代码就是固定大小)来决定了;其中注释9中的ViewGroup.LayoutParams.MATCH_PARENT 表示精确模式,大小就是窗口的大小;注释10中的 ViewGroup.LayoutParams.WRAP_CONTENT表示最大模式,但窗口大小不定,不能超过窗口的大小;注释11中的固定大小,比如为50dp,它是精确模式,大小为 LayoutParams 指定的大小。

普通的 View的 MeasureSpec 是怎么转换的呢?我们知道子 View 的 measure 过程是由父元素 ViewGroup 传递过来的,而父元素 ViewGroup 传递是在 ViewGroup 的 measureChildWithMargins方法;

c675acbd81b05e90ecc5ba277f657ebc.png

看注释12和注释13,ViewGroup 通过 getChildMeasureSpec 方法来得到子 View 的 MeasureSpec;从入参来看,子 View 的 MeasureSpec 的转换不只与父元素的 MeasureSpec 和子 View 本身的 LayoutParams 有关,

还与 子View 的 margin 及 padding 有关。

我们再具体看一下 ViewGroup 的 getChildMeasureSpec 方法;

public static int getChildMeasureSpec(int spec, int padding, int childDimension) {......//14、int size = Math.max(0, specSize - padding);......switch (specMode) {//15、case View.MeasureSpec.EXACTLY:if (childDimension >= 0) {resultSize = childDimension;resultMode = View.MeasureSpec.EXACTLY;} else if (childDimension == ViewGroup.LayoutParams.MATCH_PARENT) {// Child wants to be our size. So be it.resultSize = size;resultMode = View.MeasureSpec.EXACTLY;} else if (childDimension == ViewGroup.LayoutParams.WRAP_CONTENT) {// Child wants to determine its own size. It can't be// bigger than us.resultSize = size;resultMode = View.MeasureSpec.AT_MOST;}break;//16、case View.MeasureSpec.AT_MOST:if (childDimension >= 0) {// Child wants a specific size... so be itresultSize = childDimension;resultMode = View.MeasureSpec.EXACTLY;} else if (childDimension == ViewGroup.LayoutParams.MATCH_PARENT) {// Child wants to be our size, but our size is not fixed.// Constrain child to not be bigger than us.resultSize = size;resultMode = View.MeasureSpec.AT_MOST;} else if (childDimension == ViewGroup.LayoutParams.WRAP_CONTENT) {// Child wants to determine its own size. It can't be// bigger than us.resultSize = size;resultMode = View.MeasureSpec.AT_MOST;}break;//17、case View.MeasureSpec.UNSPECIFIED:if (childDimension >= 0) {// Child wants a specific size... let them have itresultSize = childDimension;resultMode = View.MeasureSpec.EXACTLY;} else if (childDimension == ViewGroup.LayoutParams.MATCH_PARENT) {// Child wants to be our size... find out how big it should// beresultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;resultMode = View.MeasureSpec.UNSPECIFIED;} else if (childDimension == ViewGroup.LayoutParams.WRAP_CONTENT) {// Child wants to determine its own size.... find out how// big it should beresultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;resultMode = View.MeasureSpec.UNSPECIFIED;}break;}//noinspection ResourceTypereturn View.MeasureSpec.makeMeasureSpec(resultSize, resultMode);}

看注释14,子 View 的可用大小为父元素的尺寸减去 padding。

看注释15,parentMeasureSpec 为 View.MeasureSpec.EXACTLY 模式,当 childDimension >= 0 时 就是子 View 的固定精确值(假设100px),那么子 View 的 SpecSize(值为childDimension) 就是 childSize,SpecMode 就是 View.MeasureSpec.EXACTLY;当 childDimension == ViewGroup.LayoutParams.MATCH_PARENT 时,子 View 的 SpecSize 就为父元素的大小,SpecMode 就是 View.MeasureSpec.EXACTLY;当 childDimension == ViewGroup.LayoutParams.WRAP_CONTENT 时,子 View 的 SpecSize 就是父元素的大小,SpecMode 就是 View.MeasureSpec.AT_MOST。

看注释16,parentMeasureSpec 为 View.MeasureSpec.AT_MOST 模式,当 childDimension >= 0 时 就是子 View 的固定精确值(假设100px),那么子 View 的 SpecSize(值为childDimension) 就是 childSize,SpecMode 就是 View.MeasureSpec.EXACTLY;当 childDimension == ViewGroup.LayoutParams.MATCH_PARENT 时,子 View 的 SpecSize 就为父元素的大小,SpecMode 就是 View.MeasureSpec.AT_MOST;当 childDimension == ViewGroup.LayoutParams.WRAP_CONTENT 时,子 View 的 SpecSize 就是父元素的大小,SpecMode 就是 View.MeasureSpec.AT_MOST。

看注释17,parentMeasureSpec 为 View.MeasureSpec.UNSPECIFIED 模式,当 childDimension >= 0 时 就是子 View 的固定精确值(假设100px),那么子 View 的 SpecSize(值为childDimension) 就是 childSize,SpecMode 就是 View.MeasureSpec.EXACTLY;当 childDimension == ViewGroup.LayoutParams.MATCH_PARENT 时,子 View 的 SpecSize 就为0,SpecMode 就是 View.MeasureSpec.UNSPECIFIED;当 childDimension == ViewGro-up.LayoutParams.WRAP_CONTENT 时,子 View 的 SpecSize 就是0,SpecMode 就是 View.MeasureSpec.UNSPECIFIED。

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


http://chatgpt.dhexx.cn/article/DwLtDIc3.shtml

相关文章

android Measurespec测量模式

MeasureSpecs 类 1、是一个32位的二进制数,由模式(mode)和大小(size)组成, 2、其中:32和31位代表测量模式(mode)、后30位代表测量大小(size) 3、…

MeasureSpec源码解读

文章目录 MeasureSpec的源码MeasureSpec与LayoutParams 今天来讲讲MeasureSpec吧。因为他与View的测量流程相关性很大,只有正确的理解了MeasureSpec的工作原理,我们才能更好的自定义View。那么MeasureSpec它的作用是什么呢?一般来说&#xff…

理解 MeasureSpec

在开始本篇文章之前,我们先看一段代码: Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {int expendSpec MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);super.onMeasure(widthMe…

对MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE 2, MeasureSpec.AT_MOST)的一点理解

之前 遇到ScrollView中嵌入ListView,GridView冲突的解决(让ListView全显示出来) 链接 网上查找资料,代码大致如下: import android.content.Context; import android.util.AttributeSet; import android.widget.ListV…

View的基本概念与MeasureSpec

1.基本概念 View的绘制是由measuer、layout、draw三个过程才能完整的绘制一个View,其中measure是测量View的宽、高,layout是为了确认View在父容器所在的位置,draw是负责在屏幕上将View绘制出来。View的绘制流程是从ViewRoot的performTraversa…

Android之:了解MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE 2,MeasureSpec.AT_MOST)

在自定义View和ViewGroup的时候,我们经常会遇到int型的 MeasureSpec 来表示一个组件的大小,这个变量里面不仅有组件的尺寸大小,还有大小的模式。 这个大小的模式,有点难以理解。在系统中组件的大小模式有三种: 1.精确…

Android开发 MeasureSpec介绍

搬家后的博客链接: IT客栈 www.itkezhan.org 在自定义View和ViewGroup的时候,我们经常会遇到int型的MeasureSpec来表示一个组件的大小,这个变量里面不仅有组件的尺寸大小,还有大小的模式。 这个大小的模式,有点难以理解。在系统中…

Android-测量规格(MeasureSpec)

目录 一、简介二、组成三、具体使用 一、简介 二、组成 测量规格(MeasureSpec)是由测量模式(mode)和测量大小(size)组成,共32位(int类型),其中: 测量模式(mode):占测量规格(MeasureSpec)的高2位;测量大小(size)&…

MeasureSpec学习 - 转

在自定义View和ViewGroup的时候,我们经常会遇到int型的 MeasureSpec 来表示一个组件的大小,这个变量里面不仅有组件的尺寸大小,还有大小的模式。 这个大小的模式,有点难以理解。在系统中组件的大小模式有三种: 1.精确…

MeasureSpec介绍

在自定义View和ViewGroup的时候,我们经常会遇到int型的MeasureSpec来表示一个组件的大小,这个变量里面不仅有组件的尺寸大小,还有大小的模式。 这个大小的模式,有点难以理解。在系统中组件的大小模式有三种: 1.精确模式…

Android 中MeasureSpec的创建规则

概述 在Android中,View的onMeasure()方法用来对控件进行测量,确定控件的宽高。该方法的两个参数widthMeasureSpec和heightMeasureSpec由父View计算后传入子view的measure()方法,再由子view的measure()方法传入onMeasure()方法,本…

关于google浏览器打不开网页问题之容易被忽略的点

其实google浏览器打不开 网页,原因网上有好多种,包括什么关闭防火墙、取消高级设置LAN单选框等,我也都试了,搞到最后要崩溃了,后来无意中,我输入一个http://baidu.com然后enter管用了,能打开页面…

谷歌浏览器打不开网页

今天起来发现谷歌浏览器和IE都打不开网页了,估计是我电脑代理又被修改了 在谷歌浏览器的设置--> 高级 --> 打开代理设置中 取消勾选即可修复问题。

关于谷歌浏览器打不开的解决方法

关于谷歌浏览器打不开的解决方法 打开Google,搜索,出现下面的问题,怎么解决呢,下面两种方法提供参考。 打开Google,首页显示输入网址,我们可以输入任意一个网址,例如www.baidu.com,然后就可以搜索了。 打开选项-设置…

谷歌浏览器(chrome)无法正常打开网页的解决办法

在网上看到许多新手想使用谷歌浏览器但是下载安装之后却无法打开网页 分析原因如下: 一般都是因为谷歌浏览器默认的地址栏搜索引擎为goole,由于goole属于国外的网站,我们访问是需要fan qiang才能访问的,所以无法打开网页。所以我…

chrome双击突然打不开的解决办法

这个也是没有想到,浪费了我挺长时间。我电脑的chrome突然打不开了,打不开的意思是双击了之后没有反应,但是其实是有打开进程的,这个就很坑。 网上搜了很多,什么把进程给杀掉的,重启电脑的,重装c…

Outlook 突然打不开

打开电脑正准备上班然后outlook崩了,报错建议我重装软件。。问题是现在用的都是365全家桶,也没办法单独重装一个outlook。盲试了一把repair居然修好了..再后来就经常用到它T_T..(不是什么好事) 首先有几种临时解决方法。 如果时间…

谷歌浏览器任何页面都打不开连设置也不能打开

谷歌浏览器任何页面都打不开 设置不能打开 找到谷歌右击选择属性 点击目标在最后加上 -no-sandbox即可解决 一定记得在 -no-sandbox前加上空格

谷歌浏览器网页打不开怎么办

首次下载谷歌浏览器,打开时在搜索框输入特定的网址会出现下面的界面 相信很多小伙伴也为此苦恼,今天小编为大家分享一个解决方法。 1.在其他其他浏览器搜索【极简插件】如下: 2.在右上角的搜索框里搜索【IGG】 点击推荐下载 3.将下载的压…

Chrome应用商店打不开问题

Chrome浏览器作为谷歌的亲儿子,实力方面很强。可是Chrome自从退出中国市场后一直对国内用户不太友好,但也还是有一批Chrome的忠实粉一直在用Chrome。 Chrome应用商店中有很多好玩实用的插件,我们在国内打开Chrome应用商店时是这样子的&#…