前言

  • 在了解自定义View三大流程的Measure过程前,我们需要了解一个重要基础:MeasureSpec
  • 今天,我将全面解析 MeasureSpec类的相关知识,希望你们会喜欢

Carson带你学Android自定义View文章系列:
Carson带你学Android:自定义View基础
Carson带你学Android:一文梳理自定义View工作流程
Carson带你学Android:自定义View Measure过程
Carson带你学Android:自定义View Layout过程
Carson带你学Android:自定义View Draw过程
Carson带你学Android:手把手教你写一个完整的自定义View
Carson带你学Android:Canvas类全面解析
Carson带你学Android:Path类全面解析


目录


1. 简介


2. 组成

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

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

其中,测量模式(Mode)的类型有三种


3. 具体使用

  • 测量规格(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);

4. 源码分析

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  }
}

5. 计算逻辑

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);
}

总结如下:

其中的规律总结:(以子View为标准,横向观察)

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


  • 区别于顶级View(即DecorView)的测量规格MeasureSpec计算逻辑:取决于 自身布局参数 & 窗口尺寸


6. 总结

  • 本文对自定义View绘制流程中Measure过程的基础MeasureSpec类进行了全面介绍。
  • Carson带你学Android自定义View文章系列:
    Carson带你学Android:自定义View基础
    Carson带你学Android:一文梳理自定义View工作流程
    Carson带你学Android:自定义View Measure过程
    Carson带你学Android:自定义View Layout过程
    Carson带你学Android:自定义View Draw过程
    Carson带你学Android:手把手教你写一个完整的自定义View
    Carson带你学Android:Canvas类全面解析
    Carson带你学Android:Path类全面解析

欢迎关注Carson_Ho的CSDN博客 与 公众号!

博客链接:https://carsonho.blog.csdn.net/


请帮顶 / 评论点赞!因为你的鼓励是我写作的最大动力!

Android自定义View:带你了解神秘的MeasureSpec类相关推荐

  1. [Android]自定义View带效果的滚动数字

    [Android]自定义View带效果的滚动数字 @Author GQ 2016年07月29日 一个可以让数字滚动的View,可以自定义参数,是想要的那种效果! 原文github地址 效果图 Andr ...

  2. Android 自定义view完全解析--带你通透了解自定义view

    参考转自郭霖博客带你一步步深入了解View系列 Android LayoutInflater原理分析 相信接触Android久一点的朋友对于LayoutInflater一定不会陌生,都会知道它主要是用 ...

  3. 一篇文章带你走近Android自定义view

    系列文章目录 一篇文章带你走近Android自定义view 文章目录 系列文章目录 前言 一.为什么要自定义view 二.先看看一个超级简单的自定义view(三个构造函数) 三.了解手机的坐标系 四. ...

  4. android 两边圆角,Android自定义View实现带4圆角或者2圆角的效果

    1 问题 实现任意view经过自定义带4圆角或者2圆角的效果 2 原理 1) 实现view 4圆角 我们只需要把左边的图嵌入到右边里面去,最终显示左边的图就行. 2) 实现view上2圆角 我们只需要 ...

  5. android 自定义View绘制电池电量(电池内带数字显示)

    最新公司需要一个电池内带数字的显示电池电量需求,百度了一下.参考下面这篇文章写的Android自定义View之电池电量显示. 增加了里面电池电量数字显示,还有就是一个屏幕适配.不管屏幕分辨率基本都能适 ...

  6. android自定义view获取控件,android 自定义控件View在Activity中使用findByViewId得到结果为null...

    转载:http://blog.csdn.net/xiabing082/article/details/48781489 1.  大家常常自定义view,,然后在xml 中添加该view 组件..如果在 ...

  7. Android 自定义View —— Canvas

    上一篇在android 自定义view Paint 里面 说了几种常见的Point 属性 绘制图形的时候下面总有一个canvas ,Canvas 是是画布 上面可以绘制点,线,正方形,圆,等等,需要和 ...

  8. Android自定义view详解,使用实例,自定义属性,贝塞尔曲线

    //只会触发执行onDraw方法,只会改变绘制里面的内容,条目的绘制 invalidate(); //只会触发执行onDraw方法,但是可以在子线程中刷新 postInvalidate(); //vi ...

  9. Android自定义view之基础知识

    Android自定义view之基础知识 虽然Android已经自带了很多实用的view和layout,加以调教能实现很美观的界面,但是有一些情况下,需要实现特殊的界面效果,比如我们比较熟悉的各种播放器 ...

  10. android自定义view案例,Android自定义View的实现方法实例详解

    一.自绘控件 下面我们准备来自定义一个计数器View,这个View可以响应用户的点击事件,并自动记录一共点击了多少次.新建一个CounterView继承自View,代码如下所示: 可以看到,首先我们在 ...

最新文章

  1. 词向量到Bert代码——部分习题
  2. XDP/eBPF — BPF
  3. Windows下各个盘中的文件夹属性变为隐藏,怎么取消隐藏属性
  4. 多文件的Makefile
  5. 阿里云CentOS-7.2安装mysql
  6. kdump需要开启吗_iPhone全新黑科技!用嘴玩手机!你会玩吗?
  7. 导入已有项目到svn
  8. input输入框只能输入正整数
  9. webpack基础教程:(二)
  10. 解决Mac电脑在启动时出现空白屏幕情况的解决方法
  11. 在Python里安装Jieba中文分词组件
  12. 【Pic】图片怎么去除马赛克?
  13. WINRAR5.0破解
  14. MATLAB绘制区域图形
  15. Vue2.0基本用法之组件的注册和传值(父子props,插槽,$emit)和学写购物车
  16. “科林明伦杯”哈尔滨理工大学第十届程序设计竞赛 E 赛马 python
  17. Docker系列 搭建个人云盘服务nextcloud
  18. C语言while循环语句 do while语句 for循环语句
  19. 上海市建筑标准规范合集
  20. INET_ADDRSTRLEN

热门文章

  1. Eclipse搭建Android开发环境并运行Android项目 (详细)
  2. ssh 登录linux xsell 登录Linux 提示用户密钥登录怎么解决
  3. 软考—软件设计师(软件工程基础知识)
  4. 思科路由器防火墙如何配置的方法
  5. CentOs7下Zabbix安装教程——准备工作
  6. 通信原理及系统系列7—— 什么是码间串扰
  7. Houdini工程文件合集
  8. Python三种设计模式
  9. arcgis拓扑几何,因缝隙太小而不能自动创建要素修复的处理办法
  10. C语言中图形题,c语言图形输出习题.doc