Android View 测量源码解析

Android View从设计到显示到屏幕上,共用了三大步:measure、layout、draw。今天主要讲讲View是如何测量的。

以FrameLayout为例,从measure方法开始,如图1,measure是View中final的方法,所以不能被子类重写。通过标志位mPrivateFlags去判断是否布局必须要被重新测量,关于mPrivateFlags何时赋值变化有兴趣可以找一下,这里就不阐述了。specChanged变量表明测量的条件有没有与上次测量条件发生变化,isSpecExactly变量表明测量是否是确定的(测量条件是EXACTLY,不是AT_MOST或者UNSPECIFIED),matchesSpecSize变量表明宽高的测量大小是否与现有宽高相等,最后通过specChanged、isSpecExactly、matchesSpecSize、sAlwaysRemeasureExactly四个变量确定是否需要重新测量,sAlwaysRemeasureExactly变量故名思意总是需要重新测量(当测量条件变化时),所以是否需要重新测量决定的因素是测量条件(widthMeasureSpec、heightMeasureSpec)发生变化且必须以下条件满足其一:

1、必须要重新测量,sAlwaysRemeasureExactly置为true

2、测量条件不是EXACTLY

3、大小与现有大小不一致

图1

当满足View必须要重新测量或者需要重新测量的条件下,进入if条件,如果不是强制重新测量的话,会从mMeasureCache缓存池里面去取以前测量好的,当存在以前测量好的话,直接setMeasuredDimensionRaw,测量结束,如果不存在的话,会走onMeasure方法,该方法一般都会被子类重写。文章以FrameLayout为例,故现在看看FrameLayout中onMeasure的实现。如图2,获取当前ViewGroup的子视图,当mMeasureAllChildren为true或者child.getVisibility() !=GONE(这就是为什么视图为Gone时不进行测量)时去测量每个子视图的宽高,调用measureChildWithMargins。

图2

如图3,获取子View的childWidthMeasureSpec与childHeightMeasureSpec

图3

现以获取子View宽度测量条件为例,如图4,首先拿到父视图specMode与specSize,根据父容器的specMode,做以下区分:

一、当specMode为EXACTLY:

1、当子View的childDimension>0,意思就是写死的大小,此时子View的测量mode为EXACTLY,测量大小为子View写死的大小

2、当子View大小为LayoutParams.MATCH_PARENT,此时子View的测量mode为EXACTLY,大小为父容器的大小

3、当子View的大小为LayoutParams.WRAP_CONTENT,测量mode为AT_MOST,大小为父容器的大小

图4

二、当specMode为AT_MOST:

1、当子View的childDimension>0,意思就是写死的大小,此时子View的测量mode为EXACTLY,测量大小为子View写死的大小

2、当子View大小为LayoutParams.MATCH_PARENT,此时子View的测量mode为AT_MOST,大小为父容器的大小

3、当子View的大小为LayoutParams.WRAP_CONTENT,大小为父容器的大小,测量mode为AT_MOST

三,当specMode为UNSPECIFIED这种情况很少见,故不做解析了

得到子View的测量条件后,调用child.measure(childWidthMeasureSpec, childHeightMeasureSpec),回到一开始,此时如果View没有重新onMeasure的方法,将使用View的onMeasure方法,如图5所示

图5

如图6所示,AT_MOST与EXACTLY最终拿的都是传进来的大小

图6

这样大家明白为什么自定View没有重写onMeasure时,会将父布局充满了吧,那为什么ImageView不会呢,我们看看ImageView中onMeasure方法

图7

如图7,当mDrawable为空时resizeWidth、resizeHeight都为false,不考虑padding和SuggestedMinimum大小,w、h大小都为零。此时走到widthSize =resolveSizeAndState(w, widthMeasureSpec, 0)。如图8,当specMode为AT_MOST时,此时specSize为父容器传进来的大小,size为刚刚计算的w=0,故此时大小为零,这样ImageView就不会撑大整个父布局了

图8

回到图2 frameLayout的onMeasure方法,当子View测量完了,会重新设置maxWidth、maxHeight,由于Frame Layout没有子布局的相对关系,故最大宽、高基本都是每个子View宽高的对比,到此测量完成。但发现代码没完,我们继续往下看,mMatchParentChildren size>1?

mMatchParentChildren 是什么呢,我们往上看

if (measureMatchParentChildren) {

    if (lp.width == LayoutParams.MATCH_PARENT ||

        lp.height == LayoutParams.MATCH_PARENT) {

        mMatchParentChildren.add(child);

    }

}

measureMatchParentChildren是在

final boolean measureMatchParentChildren =

MeasureSpec.getMode(widthMeasureSpec) != MeasureSpec.EXACTLY ||

MeasureSpec.getMode(heightMeasureSpec) != MeasureSpec.EXACTLY;

意思就是当父容器大小不固定时,需要把子View宽为MATCH_PARENT 或者高为MATCH_PARENT 的View放到mMatchParentChildren的集合里面去,当mMatchParentChildren大小大于1的时候,需要对这些子View重新测量,为什么这样呢?

我们可以想想,当有一个mMatchParentChildren里面的View重写了onMeasure,此时大小不是父布局分配的大小,大于父布局分配的大小,此时父布局的大小等于该View的大小,而其他mMatchParentChildren里面View的大小是父布局分配的大小,那这时这些View就没填充满整个父布局,故需要重新去测量,将这个mMatchParentChildren里面的View赋予父布局的大小,这就完美解决了

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容