Android测量

  • 最大模式(MeasureSpec.AT_MOST) 这个也就是父组件,能够给出的最大的空间,当前组件的长或宽最大只能为这么大,当然也可以比这个小。 最高两位是11的时候表示”最大模式”。即MeasureSpec.AT_MOST
  • 未指定模式(MeasureSpec.UNSPECIFIED) 这个就是说,当前组件,可以随便用空间,不受限制。 最高两位是00的时候表示”未指定模式”。即MeasureSpec.UNSPECIFIED

4.measure()

  • onMeasure()的方法调度是在measure()实现的
  • 测量结果的缓存是通过mMeasureCache 进行缓存的,结构是key-value的
  • 如果PFLAG_MEASURED_DIMENSION_SET 标记没有设置,直接抛出异常,即必须要记录View测量后的尺寸值,即调用setMeasuredDimensionRaw(xx)

源码流程

  1. 如果此次测量的结果和上次的结果不同,则表示尺寸变了,如果并且满足以下三个条件之一则需要重新布局
  2. 是否是Exactly模式
  3. 当前测量的尺寸和父布局的给的尺寸是否一致
  4. 在6.0及其以下,sAlwaysRemeasureExactly=true
  5. 如果需要重写布局则进入如下逻辑:
  6. 首先清空标记为PFLAG_MEASURED_DIMENSION_SET(测量值已记录)
  7. 尝试通过缓存中获取结果(非强制重新布局情况)
  8. 如果没有缓存则调用onMeasure()进行测量
  9. 如果有缓存则直接取缓存
  10. 记录最终的测量结果,并加入至缓存

5.ViewGroup.onMeasure()实现过程

  1. 遍历子View,调用measureChildWithMargins()
  2. 根据传入的子View,父View的测量结果,已经使用过的宽高,和子View的LayoutParams,进行处理,调用getChildMeasureSpec()获取子View的测量结果
  3. 将计算后的结果,通过MeasureSpec.makeMeasureSpec(resultSize, resultMode)进行封装
  4. 调用子View的measure()方法,将父布局测量以后的结果传递给子View,子View在onMeasure()中就可以获得MeasureSpec
  5. 通过resolveSizeAndState()方法记录测量结果

5.1 measureChildWidthMargins()

getChildMeasureSpec(int spec, int padding, int childDimension).PNG

  1. 遍历每个⼦ View,⽤ measureChildWidthMargins() 测量⼦ View

需要重写 generateLayoutParams() 并返回 MarginLayoutParams 才能使⽤measureChildWithMargins() ⽅法

有些⼦ View 可能需要重新测量(⽐如换⾏处)

测量完成后,得出⼦ View 的实际位置和尺⼨,并暂时保存

  1. 通过 getChildMeasureSpec(int spec, int padding, int childDimension) ⽅法计​
    算出⼦ View 的 widthMeasureSpec 和 heightMeasureSpec,然后调⽤ child.measure() ⽅法来让⼦ View ⾃我测量
源码流程
  1. 如果开发者写了具体值(例如 layout_width=“24dp”),就不⽤再考虑⽗View 的剩余空间了,直接⽤ LayoutParams.width / height 来作为⼦ View的限制 size,⽽限制 mode 为 EXACTLY

  2. 如果开发者写的是 MATCH_PARENT,需要根据⾃⼰的 widthMeasureSpec 或 heightMeasureSpec 中的 mode 来分情况判断

  3. 如果⾃⼰的 spec 中的 mode 是 EXACTLY 或者 AT_MOST,说明⾃⼰的尺⼨有上限,那么把 spec 中的 size 减去⾃⼰的已⽤宽度或⾼度,就是⾃⼰可以给⼦ View 的 size;

  4. 如果自己的mode 是EXACTLY,传给子View 的mode是EXACTLY

  5. 如果自己的mode 是AT_MOST,传给子View 的mode是AT_MOST

  6. 如果⾃⼰的 spec 中的 mode 是 UNSPECIFIED,说明⾃⼰的尺⼨没有上限,给⼦ View 限制的 mode 就设置为 UNSPECIFIED,size为0

  7. 如果开发者写的是 WRAP_CONTENT,即要求⼦ View 在不超限制的前提下⾃我测量,需要根据⾃⼰的 widthMeasureSpec 和 heightMeasureSpec中的 mode 来分情况判断:

  8. 如果⾃⼰的 spec 中的 mode 是 EXACTLY 或者 AT_MOST,说明⾃⼰的尺⼨有上限,那么把 spec 中的 size 减去⾃⼰的已⽤宽度或⾼度,就是⾃⼰可以给⼦ View 的尺⼨上限;⾄于 mode,就⽤AT_MOST

注意,就算⾃⼰的 mode 是 EXACTLY,传给⼦ View 的也是 AT_MOST,

  1. 如果⾃⼰的 spec 中的 mode 是 UNSPECIFIED,说明⾃⼰的尺⼨没有上限,给⼦ View 限制的 mode 就设置为 UNSPECIFIED,size为0

5.2 resolveSizeAndState()

⽤ MeasureSpec.getMode(measureSpec) 和MeasureSpec.getSize(measureSpec) 取出⽗View对⾃⼰的尺⼨限制类型和具体限制尺⼨

  1. 如果mode是EXACTLY,表示⽗ View 对⼦ View 的尺⼨做出了精确限制,所以就放弃计算出的 size,直接选⽤ measure spec 的 size
  2. 如果mode是AT_MOST,表示⽗ View 对⼦ View 的尺⼨只限制了上限。
  3. 如果计算出的 size 不⼤于 spec 中限制的 size,表示尺⼨没有超出限制,所以选⽤计算出的 size。
  4. 如果计算出的 size ⼤于 spec 中限制的 size,表示尺⼨超限了,所以选⽤spec 的 size,并且在 resolveSizeAndState() 中会添加标志MEASURED_STATE_TOO_SMALL(这个标志可以辅助⽗ View 做测量和布局的计算)
  5. 如果mode 是 UNSPECIFIED,表示⽗ View 对⼦ View 没有任何尺⼨限制,所以直接选⽤计算出的 size,忽略 spec 中的 size。

6.View.onMeasure()默认实现过程

  1. 通过getDefaultSize()对最小尺寸和父View传入的尺寸进行比较,找出合适尺寸
  2. 通过setMeasuredDimension()记录测量之后的尺寸,并通过PFLAG_MEASURED_DIMENSION_SET进行标记已经测量过

7.ViewGroup/View onMeasure()区别

  • View onMeasure() 负责将ViewGroup给的测量结果,经过一些权衡后记录测量后的尺寸值到成员变量里
  • ViewGroup onMeasure() 首先遍历测量其子布局,然后根据子布局测量结果,结合其父布局给的测量结果,经过一些权衡后记录测量后的尺寸值到成员变量里
  • 继承自ViewGroup,必须要重写onMeasure()方法才能为里边子布局测量结果

8.onMeasure() 为什么会执行多次?

  • 首先当屏幕刷新信号到来时,会执行performTraversals()方法,开启测量,布局,绘制流程

  • 在performTraversals()方法之中调用了measureHierarchy()方法用于整体测量

  • 然而在measureHierarchy()之中,有三次测量,代码实现是调用了三次performMeasure()

  • 第一次:先用预设置宽高测量ViewTree,得到测量结果

  • 第二次:如果存在比预设宽高还大的View,则不满足条件,于是设置更大的宽高,进行第二次测量

  • 第三次:发现扩大后的测量结果还不满足,则用window能拿到的最大尺寸再次测量

每次performMeasure()都会出发onMeasure()

  • measureHierarchy()执行完毕后还会调用一次performMeasure()

最后

小编这些年深知大多数初中级Android工程师,想要提升自己,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

资料⬅专栏获取
)]一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

资料⬅专栏获取

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/32457.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

1996年-2023年 全国298个地级市-外商直接投资FDI(数据收集)

外商直接投资(FDI)是一种跨国界的经济活动,它涉及外国投资者在中国境内进行的直接投资行为。这种投资行为不仅包括以货币、实物、技术等形式的资本投入,还可能包括开办独资企业、合资企业、合作企业,以及参与资源开发等…

微型操作系统内核源码详解系列五(四):cm3下svc启动任务

系列一:微型操作系统内核源码详解系列一:rtos内核源码概论篇(以freertos为例)-CSDN博客 系列二:微型操作系统内核源码详解系列二:数据结构和对象篇(以freertos为例)-CSDN博客 系列…

开发者配置项、开发者选项自定义

devOptions.vue源码 <!-- 开发者选项 &#xff08;CtrlAltShiftD&#xff09;--> <template><div :class"$options.name" v-if"visible"><el-dialog:custom-class"sg-el-dialog":append-to-body"true":close-on…

Flutter 如何发布安卓应用?

android:hardwareAccelerated“true” android:windowSoftInputMode“adjustResize”> <meta-data android:name“flutterEmbedding” android:value“2” /> Flutter生成的文件建议是大部分内容可以保留不动&#xff0c;但是可以根据需要进行修改。 具体可能要修…

STM32读写备份寄存器和实时时钟

文章目录 1. 硬件电路 2. RTC操作注意事项 操作步骤 3. 代码实现 3.1 读写备份寄存器 3.1.1 main.c 3.2 实时时钟 3.2.1 MyRTC.c 3.2.2 MyRTC.h 3.2.3 main.c 1. 硬件电路 对于BKP备份寄存器和RTC实时时钟的详细解析可以看下面这篇文章&#xff1a; STM32单片机BKP备…

Linux查看公网IP的解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

字节大牛耗时八个月又一力作,Android性能调优秘籍:设计思想与代码质量优化+程序性能优化+开发效率优化(全网疯传)

第一章、设计思想与代码质量优化 一、六大原则 二、设计模式 三、数据结构 四、算法 第二章、 程序性能优化 一、启动速度与执行效率优化 二、 布局检测与优化 三、 内存优化 四、耗电优化 五、网络传输与数据存储优化 六、APK 大小优化 第三章、 开发效率优化 一、…

双叒叕-一个-Android-MVVM-组件化架构框架?

LifecycleViewModelLiveDataViewBindingAndroid KTXOkHttp:网络请求Retrofit:网络请求MMKV:腾讯基于 mmap 内存映射的 key-value 本地存储组件Glide:快速高效的Android图片加载库ARoute:阿里用于帮助 Android App 进行组件化改造的框架 —— 支持模块间的路由、通信、解耦BaseR…

【学习笔记】CSS

CSS 1、 基础篇 1.1、选择器 1.2、长度单位 1.3、CSS2 常用属性 1.4、盒模型 1.5、浮动 1.6、定位 position2、 CSS3 2.1、新增长度单位 2.2、新增颜色表示 2.3、新增选择器 2.4、新增盒子属性 2.5、新增背景属性 …

STM32单片机BKP备份寄存器和RTC实时时钟详解

文章目录 1. Unix时间戳 2. UTC/GMT 3. 时间戳转换 4. BKP简介 5. BKP基本结构 6. RTC简介 7. RTC框架图 8. RTC基本结构 9. 代码示例 1. Unix时间戳 实时时钟&#xff0c;本质上是一个定时器&#xff0c;专门用来产生年月日时分秒。 Unix 时间戳&#xff08;Unix T…

CausalMMM:基于因果结构学习的营销组合建模

1. 摘要 在线广告中&#xff0c;营销组合建模&#xff08;Marketing Mix Modeling&#xff0c;MMM&#xff09; 被用于预测广告商家的总商品交易量&#xff08;GMV&#xff09;&#xff0c;并帮助决策者调整各种广告渠道的预算分配。传统的基于回归技术的MMM方法在复杂营销场景…

Windows10中端口被占用处理方法

前言 在Windows 10中&#xff0c;查看端口被占用情况的方法主要依赖于命令行工具netstat。以下是详细步骤&#xff0c;以及必要的解释和归纳&#xff1a; 打开命令提示符 方法1&#xff1a;使用快捷键Win R&#xff0c;打开“运行”对话框&#xff0c;输入cmd&#xff0c;然…

大疆炸机后MOV修复方法(DJI Inspire 3)

dji大疆可以说是无人机中的华为&#xff0c;产品线之广性能之高让高傲的美国人侧面&#xff0c;质量和性价比才是王道。另外产品线的细分也是制胜法宝&#xff0c;无论是手持、农用机、特殊无人机还是影视级产品DJI都有涉及&#xff0c;给人的感觉就是在无人机细分方面它已经无…

java泛型学习

没有java泛型会存在的问题 假设我们有一个方法&#xff0c;希望通过传递不同类型的参数&#xff0c;输出不同类型的对象值。正常情况下我们可能会写不同的方法来实现&#xff0c;但是这样会导致类不断增加&#xff0c;并且类方法很相似&#xff0c;不能够复用。进而导致类爆炸…

基于顺序存储的环形队列算法库构建

学习贺利坚老师基于数组的环形队列 数据结构之自建算法库——顺序环形队列_下空队列q中依次入队列数据元素abc-CSDN博客文章浏览阅读5.2k次&#xff0c;点赞6次&#xff0c;收藏6次。本文针对数据结构基础系列网络课程(3)&#xff1a;栈和队列中第9课时环形队列的存储及基本操…

华为---理解OSPF Route-ID(五)

9.5 理解OSPF Route-ID 9.5.1 原理概述 一些动态路由协议要求使用Router-ID作为路由器的身份标示&#xff0c;如果在启动这些路由协议时没有指定Router-ID,则默认使用路由器全局下的路由管理Router-ID。 Router-ID选举规则为&#xff0c;如果通过Router-ID命令配置了Router-…

1.22 LeetCode总结(基本算法)_位运算

进制的概念 进制即进位计数制&#xff0c;是利用固定的数字符号和统一的规则的带进位的计数方法。 任何一种进位计数制都有一个基数&#xff0c;基数为 X 的进位计数制称为 X 进制&#xff0c;表示每一个数位上的数运算时都是逢 X 进一。 504. 七进制数 手法1&#xff1a;当…

APP启动流程解析

简单概括启动微信的流程就是&#xff1a; 1.Launcher通知AMS 要启动微信了&#xff0c;并且告诉AMS要启动的是哪个页面也就是首页是哪个页面 2.AMS收到消息告诉Launcher知道了&#xff0c;并且把要启动的页面记下来 3.Launcher进入Paused状态&#xff0c;告诉AMS&#xff0c…

[FreeRTOS 基础知识] 互斥访问与回环队列 概念

文章目录 为什么需要互斥访问&#xff1f;使用队列实现互斥访问休眠和唤醒机制环形缓冲区 为什么需要互斥访问&#xff1f; 在裸机中&#xff0c;假设有两个函数&#xff08;func_A, func_B&#xff09;都要修改a的值&#xff08;a&#xff09;&#xff0c;那么将a定义为全局变…

css-vxe列表中ant进度条与百分比

1.vxe列表 ant进度条 <vxe-column field"actualProgress" title"进度" align"center" width"200"><template #default"{ row }"><a-progress:percent"Math.floor(row.actualProgress)"size"s…