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

Android View繪制的三大流程

移動開發(fā) Android
View的工作流程主要是指measure、layout、draw這三大流程,即測量、布局和繪制,其中measure確定View的測量寬高,layout根據(jù)測量的寬高確定View在其父View中的四個頂點的位置,而draw則將View繪制到屏幕上,這樣通過ViewGroup的遞歸遍歷,一個View樹就展現(xiàn)在屏幕上了。

介紹

View的工作流程主要是指measure、layout、draw這三大流程,即測量、布局和繪制,其中measure確定View的測量寬高,layout根據(jù)測量的寬高確定View在其父View中的四個頂點的位置,而draw則將View繪制到屏幕上,這樣通過ViewGroup的遞歸遍歷,一個View樹就展現(xiàn)在屏幕上了。說的簡單,下面帶大家一步一步從源碼中分析:

Android的View是樹形結(jié)構(gòu)的:

 

基本概念

在介紹View的三大流程之前,我們必須先介紹一些基本的概念,才能更好地理解這整個過程。

Window的概念

Window表示的是一個窗口的概念,它是站在WindowManagerService角度上的一個抽象的概念,Android中所有的視圖都是通過Window來呈現(xiàn)的,不管是Activity、Dialog還是Toast,只要有View的地方就一定有Window。

這里需要注意的是,這個抽象的Window概念和PhoneWindow這個類并不是同一個東西,PhoneWindow表示的是手機屏幕的抽象,它充當(dāng)Activity和DecorView之間的媒介,就算沒有PhoneWindow也是可以展示View的。

拋開一切,僅站在WindowManagerService的角度上,Android的界面就是由一個個Window層疊展現(xiàn)的,而Window又是一個抽象的概念,它并不是實際存在的,它是以View的形式存在,這個View就是DecorView。

關(guān)于Window這方面的內(nèi)容,我們這里先了解一個大概

DecorView的概念

DecorView是整個Window界面的最頂層View,View的測量、布局、繪制、事件分發(fā)都是由DecorView往下遍歷這個View樹。DecorView作為***View,一般情況下它內(nèi)部會包含一個豎直方向的LinearLayout,在這個LinearLayout里面有上下兩個部分(具體情況和Android的版本及主題有關(guān)),上面是【標(biāo)題欄】,下面是【內(nèi)容欄】。在Activity中我們通過setContentView所設(shè)置的布局文件其實就是被加載到【內(nèi)容欄】中的,而內(nèi)容欄的id是content,因此指定布局的方法叫setContent().

 

ViewRoot的概念

ViewRoot對應(yīng)于ViewRootImpl類,它是連接WindowManager和DecorView的紐帶,View的三大流程均是通過ViewRoot來完成的。在ActivityThread中,當(dāng)Activity對象被創(chuàng)建完之后,會講DecorView添加到Window中,同時會創(chuàng)建對應(yīng)的ViewRootImpl,并將ViewRootImpl和DecorView建立關(guān)聯(lián),并保存到WindowManagerGlobal對象中。

  1. WindowManagerGlobal.java 
  2.  
  3. root = new ViewRootImpl(view.getContext(), display);   
  4. root.setView(view, wparams, panelParentView);  

Java

View的繪制流程是從ViewRoot的performTraversals方法開始的,它經(jīng)過measure、layout和draw三個過程才能最終將一個View繪制出來,大致流程如下圖:

 

Measure測量

為了更好地理解View的測量過程,我們還需要理解MeasureSpec,它是View的一個內(nèi)部類,它表示對View的測量規(guī)格。MeasureSpec代表一個32位int值,高2位代表SpecMode(測量模式),低30位代表SpecSize(測量大小),我們可以看看它的具體實現(xiàn): 

  1. MeasureSpec.java 
  2.  
  3. public static class MeasureSpec {   
  4.         private static final int MODE_SHIFT = 30; 
  5.         private static final int MODE_MASK  = 0x3 << MODE_SHIFT; 
  6.  
  7.         /** 
  8.           * UNSPECIFIED 模式: 
  9.           * 父View不對子View有任何限制,子View需要多大就多大 
  10.           */  
  11.         public static final int UNSPECIFIED = 0 << MODE_SHIFT; 
  12.  
  13.         /** 
  14.           * EXACTYLY 模式: 
  15.           * 父View已經(jīng)測量出子Viwe所需要的精確大小,這時候View的最終大小 
  16.           * 就是SpecSize所指定的值。對應(yīng)于match_parent和精確數(shù)值這兩種模式 
  17.           */  
  18.         public static final int EXACTLY     = 1 << MODE_SHIFT; 
  19.  
  20.         /** 
  21.           * AT_MOST 模式: 
  22.           * 子View的最終大小是父View指定的SpecSize值,并且子View的大小不能大于這個值, 
  23.           * 即對應(yīng)wrap_content這種模式 
  24.           */  
  25.         public static final int AT_MOST     = 2 << MODE_SHIFT; 
  26.  
  27.         //將size和mode打包成一個32位的int型數(shù)值 
  28.         //高2位表示SpecMode,測量模式,低30位表示SpecSize,某種測量模式下的規(guī)格大小 
  29.         public static int makeMeasureSpec(int sizeint mode) { 
  30.             if (sUseBrokenMakeMeasureSpec) { 
  31.                 return size + mode; 
  32.             } else { 
  33.                 return (size & ~MODE_MASK) | (mode & MODE_MASK); 
  34.             } 
  35.         } 
  36.  
  37.         //將32位的MeasureSpec解包,返回SpecMode,測量模式 
  38.         public static int getMode(int measureSpec) { 
  39.             return (measureSpec & MODE_MASK); 
  40.         } 
  41.  
  42.         //將32位的MeasureSpec解包,返回SpecSize,某種測量模式下的規(guī)格大小 
  43.         public static int getSize(int measureSpec) { 
  44.             return (measureSpec & ~MODE_MASK); 
  45.         } 
  46.         //... 
  47.     }  

Java

MeasureSpec通過將SpecMode和SpecSize打包成一個int值來避免過多的對象內(nèi)存分配,并提供了打包和解包的方法。

SpecMode有三種類型,每一類都表示特殊的含義:

UNSPECIFIED

父容器不對View有任何限制,要多大就給多大,這種情況一般用于系統(tǒng)內(nèi)部,表示一種測量的狀態(tài);

EXACTLY

父容器已經(jīng)檢測出View所需的精確大小,這個時候View的最終打消就是SpecSize所指定的值。它對應(yīng)于LayoutParams中的match_parent和具體數(shù)值這兩種模式。

AT_MOST

父容器指定了一個可用大小即SpecSize,View的大小不能大于這個值,具體是什么值要看不同View的具體實現(xiàn)。它對應(yīng)于LayoutParams中wrap_content。

View的MeasureSpec是由父容器的MeasureSpec和自己的LayoutParams決定的,但是對于DecorView來說有點不同,因為它沒有父類。在ViewRootImpl中的measureHierarchy方法中有如下一段代碼展示了DecorView的MeasureSpec的創(chuàng)建過程,其中desiredWindowWidth和desireWindowHeight是屏幕的尺寸大?。?/p>

ViewGroup的measure

  1. childWidthMeasureSpec = getRootMeasureSpec(desiredWindowWidth, lp.width); 
  2.  
  3. childHeightMeasureSpec = getRootMeasureSpec(desiredWindowHeight, lp.height); 
  4.  
  5. performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);  

Java

再看看getRootMeasureSpec方法:

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

Java

通過以上代碼,DecorView的MeasureSpec的產(chǎn)生過程就很明確了,因為DecorView是FrameLyaout的子類,屬于ViewGroup,對于ViewGroup來說,除了完成自己的measure過程外,還會遍歷去調(diào)用所有子元素的measure方法,各個子元素再遞歸去執(zhí)行這個過程。和View不同的是,ViewGroup是一個抽象類,他沒有重寫View的onMeasure方法,這里很好理解,因為每個具體的ViewGroup實現(xiàn)類的功能是不同的,如何測量應(yīng)該讓它自己決定,比如LinearLayout和RelativeLayout。

因此在具體的ViewGroup中需要遍歷去測量子View,這里我們看看ViewGroup中提供的測量子View的measureChildWithMargins方法:

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

Java

上述方法會對子元素進行measure,在調(diào)用子元素的measure方法之前會先通過getChildMeasureSpec方法來得到子元素的MeasureSpec。從代碼上看,子元素的MeasureSpec的創(chuàng)建與父容器的MeasureSpec和本身的LayoutParams有關(guān),此外和View的margin和父類的padding有關(guān),現(xiàn)在看看getChildMeasureSpec的具體實現(xiàn):

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

Java

上述代碼根據(jù)父類的MeasureSpec和自身的LayoutParams創(chuàng)建子元素的MeasureSpec,具體過程同學(xué)們自行分析,最終的創(chuàng)建規(guī)則如下表:

 

ViewGroup在遍歷完子View后,需要根據(jù)子元素的測量結(jié)果來決定自己最終的測量大小,并調(diào)用setMeasuredDimension方法保存測量寬高值。

  1. setMeasuredDimension(resolveSizeAndState(maxWidth, widthMeasureSpec, childState),heightSizeAndState); 

Java

這里調(diào)用了resolveSizeAndState來確定最終的大小,主要是保證測量的大小不能超過父容器的***剩余空間maxWidth,這里我們看看它里面的實現(xiàn):

  1. public static int resolveSizeAndState(int sizeint measureSpec, int childMeasuredState) { 
  2.         final int specMode = MeasureSpec.getMode(measureSpec); 
  3.         final int specSize = MeasureSpec.getSize(measureSpec); 
  4.         final int result; 
  5.         switch (specMode) { 
  6.             case MeasureSpec.AT_MOST: 
  7.                 if (specSize < size) { 
  8.                     result = specSize | MEASURED_STATE_TOO_SMALL; 
  9.                 } else { 
  10.                     result = size
  11.                 } 
  12.                 break; 
  13.             case MeasureSpec.EXACTLY: 
  14.                 result = specSize; 
  15.                 break; 
  16.             case MeasureSpec.UNSPECIFIED: 
  17.             default
  18.                 result = size
  19.         } 
  20.         return result | (childMeasuredState & MEASURED_STATE_MASK); 
  21.     }  

Java

關(guān)于具體ViewGroup的onMeasure過程這里不做分析,由于每種布局的測量方式不一樣,不可能逐個分析,但在它們的onMeasure里面的步驟是有一定規(guī)律的:

1.根據(jù)各自的測量規(guī)則遍歷Children元素,調(diào)用getChildMeasureSpec方法得到Child的measureSpec;

2.調(diào)用Child的measure方法;

3.調(diào)用setMeasuredDimension確定最終的大小。

View的measure

View的measure過程由其measure方法來完成,measure方法是一個final類型的方法,這意味著子類不能重寫此方法,在View的measure方法里面會去調(diào)用onMeasure方法,我們這里只要看onMeasure的實現(xiàn)即可,如下:

  1. View.java 
  2.  
  3.     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 
  4.         setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec), 
  5.                 getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec)); 
  6.     }  

Java

代碼很簡單,我們繼續(xù)看看getDefaultSize方法的實現(xiàn):

  1. View.java 
  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.     } 

 

Java

從上述代碼可以得出,View的寬/高由specSize決定,直接繼承View的自定義控件需要重寫onMeasure方法并設(shè)置wrap_content時的自身大小,否則在布局中使用wrap_content就相當(dāng)于使用match_parent。

上述就是View的measure大致過程,在measure完成之后,通過getMeasuredWidth/Height方法就可以獲得測量后的寬高,這個寬高一般情況下就等于View的最終寬高了,因為View的layout布局的時候就是根據(jù)measureWidth/Height來設(shè)置寬高的,除非在layout中修改了measure值。

Layout布局

Layout的作用是ViewGroup用來確定子元素的位置,當(dāng)ViewGroup的位置被確定后,它在onLayout中會遍歷所有的子元素并調(diào)用其layout方法。簡單的來說就是,layout方法確定View本身的位置,而onLayout方法則會確定所有子元素的位置。

先看看View的layout方法:

  1. public void layout(int l, int t, int r, int b) { 
  2.         if ((mPrivateFlags3 & PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT) != 0) { 
  3.             onMeasure(mOldWidthMeasureSpec, mOldHeightMeasureSpec); 
  4.             mPrivateFlags3 &= ~PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT; 
  5.         } 
  6.  
  7.         int oldL = mLeft; 
  8.         int oldT = mTop; 
  9.         int oldB = mBottom; 
  10.         int oldR = mRight; 
  11.  
  12.         boolean changed = isLayoutModeOptical(mParent) ? 
  13.                 setOpticalFrame(l, t, r, b) : setFrame(l, t, r, b); 
  14.  
  15.         if (changed || (mPrivateFlags & PFLAG_LAYOUT_REQUIRED) == PFLAG_LAYOUT_REQUIRED) { 
  16.             onLayout(changed, l, t, r, b); 
  17.  
  18.             if (shouldDrawRoundScrollbar()) { 
  19.                 if(mRoundScrollbarRenderer == null) { 
  20.                     mRoundScrollbarRenderer = new RoundScrollbarRenderer(this); 
  21.                 } 
  22.             } else { 
  23.                 mRoundScrollbarRenderer = null
  24.             } 
  25.  
  26.             mPrivateFlags &= ~PFLAG_LAYOUT_REQUIRED; 
  27.  
  28.             ListenerInfo li = mListenerInfo; 
  29.             if (li != null && li.mOnLayoutChangeListeners != null) { 
  30.                 ArrayList<OnLayoutChangeListener> listenersCopy = 
  31.                         (ArrayList<OnLayoutChangeListener>)li.mOnLayoutChangeListeners.clone(); 
  32.                 int numListeners = listenersCopy.size(); 
  33.                 for (int i = 0; i < numListeners; ++i) { 
  34.                     listenersCopy.get(i).onLayoutChange(this, l, t, r, b, oldL, oldT, oldR, oldB); 
  35.                 } 
  36.             } 
  37.         } 
  38.  
  39.         mPrivateFlags &= ~PFLAG_FORCE_LAYOUT; 
  40.         mPrivateFlags3 |= PFLAG3_IS_LAID_OUT; 
  41.     }  

因微信字數(shù)限制,請點擊原文鏈接查看完整內(nèi)容

總結(jié)

到這里,View的measure、layout、draw三大流程就說完了,這里做一下總結(jié):

  • 如果是自定義ViewGroup的話,需要重寫onMeasure方法,在onMeasure方法里面遍歷測量子元素,同理onLayout方法也是一樣,***實現(xiàn)onDraw方法繪制自己;
  • 如果自定義View的話,則需要從寫onMeasure方法,處理wrap_content的情況,不需要處理onLayout,***實現(xiàn)onDraw方法繪制自己; 

 

責(zé)任編輯:龐桂玉 來源: Android開發(fā)中文站
相關(guān)推薦

2020-07-16 08:33:38

ViewGroupView

2021-09-30 07:36:51

AndroidViewDraw

2024-09-05 08:28:25

2009-07-20 16:52:18

運維管理流程北塔

2013-11-06 14:16:23

流程

2021-04-26 07:53:04

繪制流程任務(wù)

2019-04-01 14:40:24

智慧城市百度阿里

2021-09-16 06:44:04

Android進階流程

2021-09-17 06:55:50

AndroidLayoutView

2013-10-15 09:56:54

大數(shù)據(jù)

2022-07-14 23:32:25

元宇宙虛擬分身Web3.0

2013-05-20 17:07:26

2019-11-11 09:00:00

測試方案自動化測試軟件開發(fā)

2022-08-16 12:03:40

網(wǎng)絡(luò)安全硬件的安全

2020-12-19 10:51:16

勒索軟件即服務(wù)RaaS網(wǎng)絡(luò)攻擊w

2020-12-18 09:00:00

數(shù)據(jù)庫MongoDBPostgreSQL

2021-08-12 09:00:00

開發(fā)測試工具

2021-09-22 09:00:00

Python框架開發(fā)

2011-10-11 10:04:58

VMware View虛擬化

2016-11-15 16:21:26

開發(fā)編程方法
點贊
收藏

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