自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

溫故而知新 MeasureSpec在View測量中的作用

開發(fā) 前端
對于具體的View/ViewGroup 測量,就涉及到另外的一個方法measureChildWithMargins,這個方法也是在很多布局中會看到,比如LinearLayout。

[[396440]]

前言

對于MeasureSpec,你的認識有多少呢?

  • MeasureSpec是干嘛的?存在的意義在哪?
  • MeasureSpec中的mode和size到底指的是什么?
  • MeasureSpec是怎么計算的,與哪些因素有關(guān)?
  • 父View測量好子View的MeasureSpec之后,子View會怎么處理?
  • View/ViewGroup、DecorView的MeasureSpec有什么區(qū)別?
  • UNSPECIFIED這個特殊模式又有什么用呢?

介紹

首先,我們看下這個類:

  1. public static class MeasureSpec { 
  2.        private static final int MODE_SHIFT = 30; 
  3.        private static final int MODE_MASK  = 0x3 << MODE_SHIFT; 
  4.  
  5.        //00后面跟30個0 
  6.        public static final int UNSPECIFIED = 0 << MODE_SHIFT; 
  7.        //01后面跟30個0 
  8.        public static final int EXACTLY     = 1 << MODE_SHIFT; 
  9.        //10后面跟30個0 
  10.        public static final int AT_MOST     = 2 << MODE_SHIFT; 
  11.  
  12.        public static int makeMeasureSpec(int sizeint mode) { 
  13.            if (sUseBrokenMakeMeasureSpec) { 
  14.                return size + mode; 
  15.            } else { 
  16.                return (size & ~MODE_MASK) | (mode & MODE_MASK); 
  17.            } 
  18.        } 
  19.  
  20.        //獲取mode 
  21.        public static int getMode(int measureSpec) { 
  22.            //保留高2位,剩下30個0 
  23.            return (measureSpec & MODE_MASK); 
  24.        } 
  25.  
  26.        //獲取size 
  27.        public static int getSize(int measureSpec) { 
  28.         //替換高兩位00,保留低30位 
  29.            return (measureSpec & ~MODE_MASK); 
  30.        } 
  31.  
  32.    } 

我留下了比較重要的三個方法:

  • makeMeasureSpec。用于生成一個MeasureSpec,生成的方式就是size+mode,得到一個32位的int值。
  • 獲取mode。也就是取前2位的值作為mode。
  • 獲取size。也就是取后30位的值作為size。

至此,我們至少知道了MeasureSpec是一個32位的int值,高2位為mode(測量模式),低30位為size(測量大小)。

這么做的目的主要是避免過多的對象內(nèi)存分配。

所以我們可以大致猜測,這個MeasureSpec就是用來標(biāo)記View的測量參數(shù),其中測量模式可能和View具體怎么顯示有關(guān),而測量大小就是值的View實際大小。

當(dāng)然,這只是我們的初步猜測。

要搞清楚具體信息,就要從View樹的繪制測量開始說起。

DecorView的測量

上文說到,測量代碼是從ViewRootImpl的measureHierarchy開始的,然后會執(zhí)行到performMeasure方法:

  1. private void measureHierarchy(){ 
  2.  childWidthMeasureSpec = getRootMeasureSpec(baseSize, lp.width); 
  3.        childHeightMeasureSpec = getRootMeasureSpec(desiredWindowHeight, lp.height); 
  4.        performMeasure(childWidthMeasureSpec, childHeightMeasureSpec); 
  5.  
  6.  
  7.    private void performMeasure(int childWidthMeasureSpec, int childHeightMeasureSpec) { 
  8.        try { 
  9.            mView.measure(childWidthMeasureSpec, childHeightMeasureSpec); 
  10.        } finally { 
  11.            Trace.traceEnd(Trace.TRACE_TAG_VIEW); 
  12.        } 
  13.    } 

很明顯,在這里就會進行第一次MeasureSpec的計算,并且傳給了下層的mView,也就是DecorView。

那我們就來看看DecorView的MeasureSpec測量規(guī)格計算方式:

  1. private static int getRootMeasureSpec(int windowSize, int rootDimension) { 
  2.        int measureSpec; 
  3.        switch (rootDimension) { 
  4.  
  5.        case ViewGroup.LayoutParams.MATCH_PARENT: 
  6.            // Window can't resize. Force root view to be windowSize. 
  7.            measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.EXACTLY); 
  8.            break; 
  9.        case ViewGroup.LayoutParams.WRAP_CONTENT: 
  10.            // Window can resize. Set max size for root view
  11.            measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.AT_MOST); 
  12.            break; 
  13.        default
  14.            // Window wants to be an exact sizeForce root view to be that size
  15.            measureSpec = MeasureSpec.makeMeasureSpec(rootDimension, MeasureSpec.EXACTLY); 
  16.            break; 
  17.        } 
  18.        return measureSpec; 
  19.    } 

所以DecorView是和它的LayoutParams有關(guān),其實也就是跟Window的調(diào)整有關(guān),如果Window是子窗口,那么就可以調(diào)整,比如Dialog的寬高設(shè)置為WRAP_CONTENT,那么DecorView對應(yīng)的測量規(guī)格就是AT_MOST。

到此,我們也可以初步得到這個測量規(guī)格mode的含義:

  • 如果View的值是確定大小,比如MATCH_PARENT或者固定值,那么它的測量模式就是MeasureSpec.EXACTLY。
  • 如果View的值是自適應(yīng),比如WRAP_CONTENT,那么它的測量模式就是 MeasureSpec.AT_MOST。

具體是不是這樣呢?我們繼續(xù)到下層View一探究竟。

View/ViewGroup的測量

對于具體的View/ViewGroup 測量,就涉及到另外的一個方法measureChildWithMargins,這個方法也是在很多布局中會看到,比如LinearLayout。

  1. protected void measureChildWithMargins(View child, 
  2.             int parentWidthMeasureSpec, int widthUsed, 
  3.             int parentHeightMeasureSpec, int heightUsed) { 
  4.         final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams(); 
  5.  
  6.         final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec, 
  7.                 mPaddingLeft + mPaddingRight + lp.leftMargin + lp.rightMargin 
  8.                         + widthUsed, lp.width); 
  9.         final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec, 
  10.                 mPaddingTop + mPaddingBottom + lp.topMargin + lp.bottomMargin 
  11.                         + heightUsed, lp.height); 
  12.  
  13.         child.measure(childWidthMeasureSpec, childHeightMeasureSpec); 
  14.     } 

代碼不多,首先獲取子View的LayoutParams。然后根據(jù) padding、margin、width 以及 parentWidthMeasureSpec 算出寬的測量模式——childWidthMeasureSpec。

高度測量模式同理。

到此,我們的認識又前進了一步,對于子View的測量模式MeasureSpec肯定是和兩個元素有關(guān):

  • 子View的LayoutParams(包括margin,width)
  • 父View的MeasureSpec (再加上padding)

繼續(xù)看看getChildMeasureSpec方法:

  1. public static int getChildMeasureSpec(int spec, int padding, int childDimension) { 
  2.         int specMode = MeasureSpec.getMode(spec); 
  3.         int specSize = MeasureSpec.getSize(spec); 
  4.  
  5.         int size = Math.max(0, specSize - padding); 
  6.  
  7.         int resultSize = 0; 
  8.         int resultMode = 0; 
  9.  
  10.         switch (specMode) { 
  11.         // Parent has imposed an exact size on us 
  12.         case MeasureSpec.EXACTLY: 
  13.             if (childDimension >= 0) { 
  14.                 resultSize = childDimension; 
  15.                 resultMode = MeasureSpec.EXACTLY; 
  16.             } else if (childDimension == LayoutParams.MATCH_PARENT) { 
  17.                 // Child wants to be our size. So be it. 
  18.                 resultSize = size
  19.                 resultMode = MeasureSpec.EXACTLY; 
  20.             } else if (childDimension == LayoutParams.WRAP_CONTENT) { 
  21.                 // Child wants to determine its own size. It can't be 
  22.                 // bigger than us. 
  23.                 resultSize = size
  24.                 resultMode = MeasureSpec.AT_MOST; 
  25.             } 
  26.             break; 
  27.  
  28.         // Parent has imposed a maximum size on us 
  29.         case MeasureSpec.AT_MOST: 
  30.             if (childDimension >= 0) { 
  31.                 // Child wants a specific size... so be it 
  32.                 resultSize = childDimension; 
  33.                 resultMode = MeasureSpec.EXACTLY; 
  34.             } else if (childDimension == LayoutParams.MATCH_PARENT) { 
  35.                 // Child wants to be our size, but our size is not fixed. 
  36.                 // Constrain child to not be bigger than us. 
  37.                 resultSize = size
  38.                 resultMode = MeasureSpec.AT_MOST; 
  39.             } else if (childDimension == LayoutParams.WRAP_CONTENT) { 
  40.                 // Child wants to determine its own size. It can't be 
  41.                 // bigger than us. 
  42.                 resultSize = size
  43.                 resultMode = MeasureSpec.AT_MOST; 
  44.             } 
  45.             break; 
  46.  
  47.         // Parent asked to see how big we want to be 
  48.         case MeasureSpec.UNSPECIFIED: 
  49.             if (childDimension >= 0) { 
  50.                 // Child wants a specific size... let him have it 
  51.                 resultSize = childDimension; 
  52.                 resultMode = MeasureSpec.EXACTLY; 
  53.             } else if (childDimension == LayoutParams.MATCH_PARENT) { 
  54.                 // Child wants to be our size... find out how big it should 
  55.                 // be 
  56.                 resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size
  57.                 resultMode = MeasureSpec.UNSPECIFIED; 
  58.             } else if (childDimension == LayoutParams.WRAP_CONTENT) { 
  59.                 // Child wants to determine its own size.... find out how 
  60.                 // big it should be 
  61.                 resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size
  62.                 resultMode = MeasureSpec.UNSPECIFIED; 
  63.             } 
  64.             break; 
  65.         } 
  66.         //noinspection ResourceType 
  67.         return MeasureSpec.makeMeasureSpec(resultSize, resultMode); 
  68.     } 

代碼其實很簡單,就是對子View的LayoutParams和父View的specMode、specSize,共同計算出子View的MeasureSpec。

舉其中一個例子,當(dāng)父view的測量模式為MeasureSpec.EXACTLY,子View寬的LayoutParams為MATCH_PARENT。想象一下,這種情況,子View的寬肯定就會占滿父View的大小,所以子View的測量模式中的mode肯定就是確定值,為MeasureSpec.EXACTLY,而大小就是父View的大小了。對應(yīng)的代碼就是:

  1. case MeasureSpec.AT_MOST: 
  2. if (childDimension == LayoutParams.MATCH_PARENT) { 
  3.     // Child wants to be our size. So be it. 
  4.     resultSize = size
  5.     resultMode = MeasureSpec.EXACTLY; 
  6. }  

綜合所有的情況,很經(jīng)典的一張表格就來了:

這里我們也可以明確了MeasureSpec中mode的含義:

  • MeasureSpec.EXACTLY。父View可以確定子View的精確大小,比如子View大小是固定的值,在所有的情況下都會是EXACTLY模式。
  • MeasureSpec.AT_MOST。父View給定一個最大的值,意思是子View大小可以不確定,但是肯定不能超過某個最大的值,例如窗口的大小。
  • MeasureSpec.UNSPECIFIED。父View對子View完全沒限制,要多大給多大。這個模式似乎聽起來有點奇怪?待會我們再細談。

到此,似乎就結(jié)束了?當(dāng)然沒啦,獲取子View的MeasureSpec之后,子View又會怎么處理呢?

View對于MeasureSpec的處理

繼續(xù)上文,測量子View的測量規(guī)格之后,會調(diào)用child.measure方法。

  1. protected void measureChildWithMargins() { 
  2.         final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams(); 
  3.  
  4.         final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec, 
  5.                 mPaddingLeft + mPaddingRight + lp.leftMargin + lp.rightMargin 
  6.                         + widthUsed, lp.width); 
  7.         final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec, 
  8.                 mPaddingTop + mPaddingBottom + lp.topMargin + lp.bottomMargin 
  9.                         + heightUsed, lp.height); 
  10.  
  11.         child.measure(childWidthMeasureSpec, childHeightMeasureSpec); 
  12.     } 
  13.  
  14. public final void measure(int widthMeasureSpec, int heightMeasureSpec) { 
  15.  onMeasure(widthMeasureSpec, heightMeasureSpec); 
  16.      

child.measure方法也就是View的measure方法,也就是走到了onMeasure方法,繼續(xù)看看:

  1. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 
  2.      setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec), 
  3.              getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec)); 
  4.  } 
  5.  
  6.  protected final void setMeasuredDimension(int measuredWidth, int measuredHeight) { 
  7.      if (optical != isLayoutModeOptical(mParent)) { 
  8.          measuredWidth  += optical ? opticalWidth  : -opticalWidth; 
  9.          measuredHeight += optical ? opticalHeight : -opticalHeight; 
  10.      } 
  11.      setMeasuredDimensionRaw(measuredWidth, measuredHeight); 
  12.  } 

哦~最后原來是給子View的measuredWidth和measuredHeight賦值了,所賦的值就是getDefaultSize方法返回的大小。

而這個measuredWidth是干嘛的呢?搜索一下:

  1. public final int getMeasuredWidth() { 
  2.  //MEASURED_SIZE_MASK用于限制大小的 
  3.         return mMeasuredWidth & MEASURED_SIZE_MASK; 
  4.     } 

這不就是我們獲取view的大小調(diào)用的方法嗎?所以小結(jié)一下:

  • 父view通過父View的MeasureSpec和子View的LayoutParams算出了子View的MeasureSpec。
  • 然后子View通過MeasureSpec計算了measuredWidth
  • 而這個measuredWidth也就是我們可以獲取View寬高所調(diào)用的方法。

最后就是看看getDefaultSize方法干了啥,也就是驗證MeasureSpec中size是不是就是我們要獲取的View的寬高呢?

  1. getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec) 
  2.  
  3.    public static int getDefaultSize(int sizeint measureSpec) { 
  4.        int result = size
  5.        int specMode = MeasureSpec.getMode(measureSpec); 
  6.        int specSize = MeasureSpec.getSize(measureSpec); 
  7.  
  8.        switch (specMode) { 
  9.        case MeasureSpec.UNSPECIFIED: 
  10.            result = size
  11.            break; 
  12.        case MeasureSpec.AT_MOST: 
  13.        case MeasureSpec.EXACTLY: 
  14.            result = specSize; 
  15.            break; 
  16.        } 
  17.        return result; 
  18.    } 

可以看到,在AT_MOST和EXACTLY這兩種常用的情況下,確實是等于測量大小specSize的。

只是在一個特殊情況,也就是UNSPECIFIED的時候,這個大小會等于getSuggestedMinimumWidth()方法的大小。

問題來了,UNSPECIFIED模式到底是啥,getSuggestedMinimumWidth()方法又做了什么?

UNSPECIFIED

很多文章會忽略這個模式,其實它也是很重要的,在前兩天的討論群中,我們還討論了這個問題,一起看看吧~

首先,我們看看什么時候會存在UNSPECIFIED模式呢?它的概念是父View對子View的大小沒有限制,很容易想到的一個控件就是ScrollView,那么在ScrollView中肯定有對這個模式的設(shè)置:

  1. @Override 
  2.    protected void measureChildWithMargins(View child, int parentWidthMeasureSpec, int widthUsed, 
  3.            int parentHeightMeasureSpec, int heightUsed) { 
  4.        final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams(); 
  5.  
  6.        final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec, 
  7.                mPaddingLeft + mPaddingRight + lp.leftMargin + lp.rightMargin 
  8.                        + widthUsed, lp.width); 
  9.        final int usedTotal = mPaddingTop + mPaddingBottom + lp.topMargin + lp.bottomMargin + 
  10.                heightUsed; 
  11.        final int childHeightMeasureSpec = MeasureSpec.makeSafeMeasureSpec( 
  12.                Math.max(0, MeasureSpec.getSize(parentHeightMeasureSpec) - usedTotal), 
  13.                MeasureSpec.UNSPECIFIED); 
  14.  
  15.        child.measure(childWidthMeasureSpec, childHeightMeasureSpec); 
  16.    } 

沒錯,在ScrollView中重寫了measureChildWithMargins方法,比對下剛才ViewGroup的measureChildWithMargins方法,發(fā)現(xiàn)有什么不對了嗎?

childWidthMeasureSpec的計算沒有什么變化,還是調(diào)用了getChildMeasureSpec方法,但是childHeightMeasureSpec不對勁了,直接調(diào)用了makeSafeMeasureSpec方法生成了MeasureSpec,而且!而且!直接把SpecMode設(shè)置成了MeasureSpec.UNSPECIFIED。

也就是對于子View的高度是無限制的,這也符合ScrollView的理念。

所以當(dāng)ScrollView嵌套一個普通View的時候,就會觸發(fā)剛才getDefaultSize中UNSPECIFIED的邏輯,也就是View的實際大小為getSuggestedMinimumWidth的大小。

繼續(xù)看看getSuggestedMinimumWidth到底獲取的是什么大?。?/p>

  1. protected int getSuggestedMinimumWidth() { 
  2.         return (mBackground == null) ? mMinWidth : max(mMinWidth, mBackground.getMinimumWidth()); 
  3.     } 

就一句代碼:

  • 如果view的背景為null,則等于最小寬度mMinWidth。
  • 如果view的背景不為null,則等于最小寬度和 背景的最小寬度 中取較大值。

所以如果View沒有設(shè)置背景,沒有設(shè)置mMinWidth,那么ScrollView嵌套View的情況,View的寬度就是為0,即使設(shè)置了固定值也沒用。

這只是UNSPECIFIED在普通View中的處理情況,不同的情況對UNSPECIFIED的處理方式都不一樣,比如TextView、RecycleView等等。

下次會專門出一篇UNSPECIFIED的文章,到時候見。

總結(jié)

今天回顧了MeasureSpec的相關(guān)知識點:

  • MeasureSpec的基本概念:
  • MeasureSpec為一個32位的int值。
  • SpecMode為高兩位,一共三種模式,代表父View對子View的大小限制模式,比如最大可用大小——AT_MOST。

SpecSize為低30位,代表父View給子View測量好的寬高。這個寬高大概率等于View的實際寬高,但是也有例外情況,也就是UNSPECIFIED的情況。

測量流程中的MeasureSpec:

  • View輸?shù)臏y量流程開始于ViewRootImpl的measureHierarchy,也是在這里開始了第一次MeasureSpec的計算。
  • 第一次MeasureSpec的計算也就是DecorView的MeasureSpec計算,是通過自身的LayoutParams相關(guān),也就是和Window大小有關(guān)。
  • 然后就開始子View/ViewGroup的MeasureSpec計算,是通過父View的MeasureSpec和子View的LayoutParams相關(guān)。
  • 計算完子View的MeasureSpec之后,就開始調(diào)用onMeasure方法,計算出View的實際大小。
  • 如果是UNSPECIFIED模式,實際大小為。否則實際大小就等于計算好的specSize。

參考

《Android開發(fā)藝術(shù)探索》

本文轉(zhuǎn)載自微信公眾號「碼上積木」,作者積木zz。轉(zhuǎn)載本文請聯(lián)系碼上積木公眾號。

 

責(zé)任編輯:武曉燕 來源: 碼上積木
相關(guān)推薦

2012-03-26 10:12:25

C#

2022-06-30 08:01:33

ProxyReflecthandler

2019-06-18 10:31:23

數(shù)據(jù)庫端口URL

2019-06-19 08:14:14

數(shù)據(jù)庫驅(qū)動URL

2021-05-29 10:11:00

Kafa數(shù)據(jù)業(yè)務(wù)

2021-05-31 07:44:08

Kafka分布式系統(tǒng)

2016-12-14 15:13:30

GradleAndroid定制化打包

2019-04-23 16:19:01

網(wǎng)絡(luò)命令網(wǎng)絡(luò)故障ping

2023-08-01 14:36:00

JavaScript開發(fā)

2021-05-07 10:20:11

前端開發(fā)技術(shù)

2020-07-16 08:04:21

瀏覽器緩存策略

2016-12-30 13:52:55

網(wǎng)絡(luò)事件

2024-05-13 10:27:37

DevOps云技術(shù)IT

2020-12-01 00:03:16

新冠疫情分析數(shù)字化

2019-01-14 15:31:42

HTTP23

2010-05-05 10:35:50

CIO

2023-04-02 23:22:04

GPU流水線體系

2022-11-15 07:30:04

EverDB云端alive

2010-05-25 16:04:45

VoIPVoWLAN

2021-08-06 10:36:15

首席數(shù)據(jù)官CDO數(shù)字化轉(zhuǎn)型
點贊
收藏

51CTO技術(shù)棧公眾號