Android-测量规格(MeasureSpec)

article/2025/8/25 22:08:53

目录

  • 一、简介
  • 二、组成
  • 三、具体使用

一、简介

在这里插入图片描述

二、组成

测量规格(MeasureSpec)是由测量模式(mode)和测量大小(size)组成,共32位(int类型),其中:

  • 测量模式(mode):占测量规格(MeasureSpec)的高2位;
  • 测量大小(size):占测量规格(MeasureSpec)的低30位。

在这里插入图片描述
其中,测量模式(Mode)的类型有三种

在这里插入图片描述

三、具体使用

  • 测量规格(MeasureSpec)的封装类是:MeasureSpec类
  • MeasureSpec类用一个变量封装了测量模式(mode)和测量大小(size):通过使用二进制,将测量模式(mode)和测量大小(size)打包成一个int值,并提供了打包和解包的方法,这样的做法是为了减少对象内存分配和提高存取效率。具体使用如下所示:
// 1. 获取测量模式(Mode)
int specMode = MeasureSpec.getMode(measureSpec)// 2. 获取测量大小(Size)
int specSize = MeasureSpec.getSize(measureSpec)// 3. 通过Mode 和 Size 生成新的SpecMode
int measureSpec=MeasureSpec.makeMeasureSpec(size, mode);

四、源码分析

public class MeasureSpec {// 进位大小 = 2的30次方// int的大小为32位,所以进位30位 = 使用int的32和31位做标志位private static final int MODE_SHIFT = 30;  // 运算遮罩:0x3为16进制,10进制为3,二进制为11// 3向左进位30 = 11 00000000000(11后跟30个0)  // 作用:用1标注需要的值,0标注不要的值。因1与任何数做与运算都得任何数、0与任何数做与运算都得0private static final int MODE_MASK  = 0x3 << MODE_SHIFT;  // UNSPECIFIED的模式设置:0向左进位30 = 00后跟30个0,即00 00000000000// 通过高2位public static final int UNSPECIFIED = 0 << MODE_SHIFT;  // EXACTLY的模式设置:1向左进位30 = 01后跟30个0 ,即01 00000000000public static final int EXACTLY = 1 << MODE_SHIFT;  // AT_MOST的模式设置:2向左进位30 = 10后跟30个0,即10 00000000000public static final int AT_MOST = 2 << MODE_SHIFT;  /*** makeMeasureSpec()方法* 作用:根据提供的size和mode得到一个详细的测量结果吗,即measureSpec**/ public static int makeMeasureSpec(int size, int mode) {  return size + mode;  // measureSpec = size + mode;此为二进制的加法 而不是十进制// 设计目的:使用一个32位的二进制数,其中:32和31位代表测量模式(mode)、后30位代表测量大小(size)// 例如size=100(4),mode=AT_MOST,则measureSpec=100+10000...00=10000..00100  }  /*** getMode()方法* 作用:通过measureSpec获得测量模式(mode)**/    public static int getMode(int measureSpec) {  return (measureSpec & MODE_MASK);  // 即:测量模式(mode) = measureSpec & MODE_MASK;  // MODE_MASK = 运算遮罩 = 11 00000000000(11后跟30个0)//原理:保留measureSpec的高2位(即测量模式)、使用0替换后30位// 例如10 00..00100 & 11 00..00(11后跟30个0) = 10 00..00(AT_MOST),这样就得到了mode的值}  /*** getSize方法* 作用:通过measureSpec获得测量大小size**/       public static int getSize(int measureSpec) {  return (measureSpec & ~MODE_MASK);  // size = measureSpec & ~MODE_MASK;  // 原理类似上面,即 将MODE_MASK取反,也就是变成了00 111111(00后跟30个1),将32,31替换成0也就是去掉mode,保留后30位的size  } 
} 

五、计算逻辑
View的MeasureSpec值计算取决于两个因素:

  • View自身的布局参数(LayoutParams)
  • 父容器的测量规格(MeasureSpec)

即View的大小是由自身布局参数(LayoutParams)和父容器的测量规格(MeasureSpec)共同决定的。

MeasureSpec值的具体计算逻辑封装在getChildMeasureSpec()里,具体计算逻辑如下源码所示。

/*** 源码分析:getChildMeasureSpec()* 作用:根据父视图的MeasureSpec & 布局参数LayoutParams,计算单个子View的MeasureSpec* 注:子view的大小由父view的MeasureSpec值 和 子view的LayoutParams属性 共同决定**/public static int getChildMeasureSpec(int spec, int padding, int childDimension) {  // 参数说明// * @param spec 父view的详细测量值(MeasureSpec) // * @param padding view当前尺寸的的内边距和外边距(padding,margin) // * @param childDimension 子视图的布局参数(宽/高)//父view的测量模式int specMode = MeasureSpec.getMode(spec);     //父view的大小int specSize = MeasureSpec.getSize(spec);     //通过父view计算出的子view = 父大小-边距(父要求的大小,但子view不一定用这个值)   int size = Math.max(0, specSize - padding);  //子view想要的实际大小和模式(需要计算)  int resultSize = 0;  int resultMode = 0;  //通过父view的MeasureSpec和子view的LayoutParams确定子view的大小  // 当父view的模式为EXACITY时,父view强加给子view确切的值//一般是父view设置为match_parent或者固定值的ViewGroup switch (specMode) {  case MeasureSpec.EXACTLY:  // 当子view的LayoutParams>0,即有确切的值  if (childDimension >= 0) {  //子view大小为子自身所赋的值,模式大小为EXACTLY  resultSize = childDimension;  resultMode = MeasureSpec.EXACTLY;  // 当子view的LayoutParams为MATCH_PARENT时(-1)  } else if (childDimension == LayoutParams.MATCH_PARENT) {  //子view大小为父view大小,模式为EXACTLY  resultSize = size;  resultMode = MeasureSpec.EXACTLY;  // 当子view的LayoutParams为WRAP_CONTENT时(-2)      } else if (childDimension == LayoutParams.WRAP_CONTENT) {  //子view决定自己的大小,但最大不能超过父view,模式为AT_MOST  resultSize = size;  resultMode = MeasureSpec.AT_MOST;  }  break;  // 当父view的模式为AT_MOST时,父view强加给子view一个最大的值。(一般是父view设置为wrap_content)  case MeasureSpec.AT_MOST:  // 道理同上  if (childDimension >= 0) {  resultSize = childDimension;  resultMode = MeasureSpec.EXACTLY;  } else if (childDimension == LayoutParams.MATCH_PARENT) {  resultSize = size;  resultMode = MeasureSpec.AT_MOST;  } else if (childDimension == LayoutParams.WRAP_CONTENT) {  resultSize = size;  resultMode = MeasureSpec.AT_MOST;  }  break;  // 当父view的模式为UNSPECIFIED时,父容器不对view有任何限制,要多大给多大// 多见于ListView、GridView  case MeasureSpec.UNSPECIFIED:  if (childDimension >= 0) {  // 子view大小为子自身所赋的值  resultSize = childDimension;  resultMode = MeasureSpec.EXACTLY;  } else if (childDimension == LayoutParams.MATCH_PARENT) {  // 因为父view为UNSPECIFIED,所以MATCH_PARENT的话子类大小为0  resultSize = 0;  resultMode = MeasureSpec.UNSPECIFIED;  } else if (childDimension == LayoutParams.WRAP_CONTENT) {  // 因为父view为UNSPECIFIED,所以WRAP_CONTENT的话子类大小为0  resultSize = 0;  resultMode = MeasureSpec.UNSPECIFIED;  }  break;  }  return MeasureSpec.makeMeasureSpec(resultSize, resultMode);  
}  

总结如下:
在这里插入图片描述

由于UNSPECIFIED模式适用于系统内部多次measure情况,很少用到,故此处不讨论

其中的规律总结:(以子View为标准,横向观察)
在这里插入图片描述

区别于顶级View(即DecorView)的测量规格MeasureSpec计算逻辑:取决于 自身布局参数 & 窗口尺寸
在这里插入图片描述


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

相关文章

MeasureSpec学习 - 转

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

MeasureSpec介绍

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

Android 中MeasureSpec的创建规则

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

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

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

谷歌浏览器打不开网页

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

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

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

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

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

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

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

Outlook 突然打不开

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

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

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

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

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

Chrome应用商店打不开问题

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

google打不开?更改一下chrome设置,畅通无阻玩谷歌

最近不知道是怎么回事&#xff0c;google&#xff08;谷歌&#xff09;一直打不开&#xff0c;gmail也是时断时续&#xff0c;十分痛苦。google服务对于的重要性不言而喻&#xff0c;特别是做 一些国外的项目&#xff0c;google的服务中断会严重影响收益。其实&#xff0c;只要…

sql 抛出异常raiserror()

说明 用于抛出一个异常或错误。这个错误可以被程序捕捉到。 实例 declare error_mes varchar(1000) set error_mes1314520886的ERP_ICStockBillEntry中间表数据的收料仓库编码不存在于系统中 raiserror(error_mes,13,1,张三)输出

SQL 中 RAISERROR 的用法

raiserror 是由单词 raise error 组成 raise 增加; 提高; 提升 raiserror 的作用 &#xff1a; raiserror 是用于抛出一个错误。[ 以下资料来源于sql server 2005的帮助 ] 其语法如下&#xff1a; RAISERROR ( { msg_id | msg_str | local_variable } …

RAISERROR (Transact-SQL)

从msdn上找到的&#xff0c;很详细了生成错误消息并启动会话的错误处理。RAISERROR 可以引用 sys.messages 目录视图中存储的用户定义消息&#xff0c;也可以动态建立消息。该消息作为服务器错误消息返回到调用应用程序&#xff0c;或返回到 TRY…CATCH 构造的关联 CATCH 块。 …

RAISE_APPLICATION_ERROR 用法

可能不是很多人知道 RAISE_APPLICATION_ERROR 的用途是什么&#xff0c;虽然从字面上已经猜到这个函数是干什么用的。平时用来测试的异常处理 我们都是通过dbms_output.put_line来输出异常信息&#xff0c;但是在实际的应用中&#xff0c;需要把异常信息返回给调用的客户端。 …

RAISERROR 的用法(转)

raiserror 的作用&#xff1a; raiserror 是用于抛出一个错误。[ 以下资料来源于sql server 2005的帮助 ] 其语法如下&#xff1a; RAISERROR ( { msg_id | msg_str | local_variable } { ,severity ,state } [ ,argument [ ,...n ] ] ) [ WITH option…

raiserror的用法

描述&#xff1a;raiserror &#xff1a;是用于抛出一个错误 第一个参数&#xff1a;{ msg_id | msg_str | local_variable } msg_id&#xff1a;表示可以是一个sys.messages表中定义的消息代号&#xff1b; 使用 sp_addmessage 存储在 sys.messages 目录视图中的用户定义错误消…

strerror函数

strerror是一个库函数&#xff0c;这个函数的功能就是将errno转换为方便我们理解的字符串信息。我们可以在linux终端命令行中输入“man 3 strerror”来查看这个库函数的详细信息&#xff0c;如下图所示。 从上图我们可以知道&#xff0c;该库函数的原型是&#xff1a;char *st…