欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android view繪制流程詳解

 更新時(shí)間:2021年05月24日 09:29:04   作者:佛系編碼  
View 的繪制是在 ViewRoot 的 performTraversals() 開(kāi)始的,它歷經(jīng) measure(測(cè)量), layout(布局), draw(繪制) 三個(gè)流程將 View 顯示在屏幕上。

繪制流程

  • measure 流程測(cè)量出 View 的寬高尺寸。
  • layout 流程確定 View 的位置及最終尺寸。
  • draw 流程將 View 繪制在屏幕上。

Measure 測(cè)量流程

系統(tǒng)是通過(guò) MeasureSpec 測(cè)量 View 的,在了解測(cè)量過(guò)程之前一定要了解這個(gè) MeasureSpec 。

MeasureSpec

MeasureSpec 是一個(gè) 32 位的 int 值打包而來(lái)的,打包為 MeasureSpec 主要是為了避免過(guò)多的對(duì)象內(nèi)存分配。

為了方便操作,MeasureSpec 提供了快捷的打包和解包的快捷方法。

  • MeasureSpec.makeMeasureSpec( int size, int mode)
  • MeasureSpec.getMode(int measureSpec)
  • MeasureSpec.getSize(int measureSpec)

MeasureSpec 其中前 2 位表示測(cè)量的模式 SpecMode,后邊 30 位表示某種測(cè)量模式下的尺寸 SpecSize。

MeasureSpec 中有三種測(cè)量模式

  • UNSPECIFIED 不指定具體尺寸,完全由 View 自己發(fā)揮。
  • EXACTLY 精確模式,這種模式下使用后邊的 specSize ,一般對(duì)應(yīng)于 LayoutParams 的 match_content 和設(shè)置的精確尺寸。
  • AT_MOST 最大模式,這種模式下 view 的最大尺寸不能超過(guò)后邊的 specSize ,一般對(duì)應(yīng)于 LayoutParams 的 wrap_content

在測(cè)量 View 的時(shí)候,系統(tǒng)會(huì)將自己的 LayoutParams 參數(shù)在父容器的 MeasureSpec 影響下轉(zhuǎn)換為自己的MeasureSpec ,然后再通過(guò)這個(gè) MeasureSpec 測(cè)量自身的寬高。

需要注意的是View 的MeasureSpec 不是唯一由 LayoutParams 決定的,是在父容器的共同影響下創(chuàng)建來(lái)的。

在 ViewGroup 的 measureChild() 可以看到具體的實(shí)現(xiàn)思路,getChildMeasureSpec() 里就是將 layoutParams 轉(zhuǎn)換為 measureSpec 的實(shí)現(xiàn)思路。

protected void measureChild(View child, int parentWidthMeasureSpec,
        int parentHeightMeasureSpec) {
        //拿到子元素的 LayoutParams 參數(shù)
    final LayoutParams lp = child.getLayoutParams();

    //創(chuàng)建子元素的 measureSpec 
    final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
            mPaddingLeft + mPaddingRight, lp.width);
    final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,
            mPaddingTop + mPaddingBottom, lp.height);

    //將測(cè)量傳遞到子元素
    child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}

public static int getChildMeasureSpec(int spec, int padding, int childDimension) {
    //解析父容器的 measureSpec ,解析出模式和尺寸
    int specMode = MeasureSpec.getMode(spec);
    int specSize = MeasureSpec.getSize(spec);

    int size = Math.max(0, specSize - padding);

    int resultSize = 0;
    int resultMode = 0;

    switch (specMode) {
    // 父容器是精確模式的情況,設(shè)置了精確尺寸。
    case MeasureSpec.EXACTLY:
        if (childDimension >= 0) {
        //子元素本身是設(shè)置的精確尺寸,就是EXACTLY 模式,尺寸就是設(shè)置的尺寸。
            resultSize = childDimension;
            resultMode = MeasureSpec.EXACTLY;
        } else if (childDimension == LayoutParams.MATCH_PARENT) {
            // 子元素設(shè)置的 match_content 充滿入容器,就把尺寸設(shè)置為入容器的尺寸,模式設(shè)置為EXACTLY
            resultSize = size;
            resultMode = MeasureSpec.EXACTLY;
        } else if (childDimension == LayoutParams.WRAP_CONTENT) {
            // 包裹模式下,子元素可以自己設(shè)置尺寸,但是不能超過(guò)夫容器的尺寸。模式為AT_MOST,尺寸為父容器的尺寸。
            resultSize = size;
            resultMode = MeasureSpec.AT_MOST;
        }
        break;

    //父容器是最大模式
    case MeasureSpec.AT_MOST:
        if (childDimension >= 0) {
            // 設(shè)置為子元素的尺寸,為精確模式
            resultSize = childDimension;
            resultMode = MeasureSpec.EXACTLY;
        } else if (childDimension == LayoutParams.MATCH_PARENT) {
            // 子元素想充滿父容器,應(yīng)該設(shè)置為父容器的尺寸,但是父容器是最大模式,沒(méi)有精確尺寸。
            // 所以將子元素設(shè)置為最大模式,不能超過(guò)父容器目前的尺寸。
            resultSize = size;
            resultMode = MeasureSpec.AT_MOST;
        } else if (childDimension == LayoutParams.WRAP_CONTENT) {
            // 子元素沒(méi)有精確尺寸,想包裹自身,這種模式下,設(shè)置為最大模式,不超過(guò)父容器尺寸就好。
            // bigger than us.
            resultSize = size;
            resultMode = MeasureSpec.AT_MOST;
        }
        break;

    // 父容器沒(méi)有限制,子元素自己發(fā)揮
    case MeasureSpec.UNSPECIFIED:
        if (childDimension >= 0) {
            //子元素自己有設(shè)置的值,就好實(shí)用自己的值,設(shè)置為精確模式
            resultSize = childDimension;
            resultMode = MeasureSpec.EXACTLY;
        } else if (childDimension == LayoutParams.MATCH_PARENT) {
            // 子元素想充滿父容器,那就找到父容器的尺寸,但父容器的尺寸未知,還是要自己發(fā)揮 UNSPECIFIED。
            resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
            resultMode = MeasureSpec.UNSPECIFIED;
        } else if (childDimension == LayoutParams.WRAP_CONTENT) {
            // 只元素是包裹自身,父容器無(wú)法給出參考,所以讓子元素自己去隨意發(fā)揮,仍然是UNSPECIFIED
            resultSize = View.sUseZeroUnspecifiedMeasureSpec ? 0 : size;
            resultMode = MeasureSpec.UNSPECIFIED;
        }
        break;
    }
    //使用打包方法,將子元素的模式和尺寸打包并返回
    return MeasureSpec.makeMeasureSpec(resultSize, resultMode);
}

measure 流程是在 ViewRoot 的 performMeasure() 里開(kāi)始的。

在這里會(huì)將 DecorView 的 layoutParams 在 window 的 measureSpec 影響下轉(zhuǎn)換為自己的 measureSpec 。 然后調(diào)用 DecorView 的 measure() 將寬高的 measureSpec 傳入,在 measure() 里,decorView 開(kāi)始自己的測(cè)量。

從 DecorView 的 measure() 開(kāi)始,整個(gè) View 樹(shù)的測(cè)量流程就開(kāi)始了。

View 的測(cè)量都是在 measure() 里進(jìn)行的,這是個(gè) final 類型的方法,里面的實(shí)現(xiàn)比較簡(jiǎn)單會(huì)有一些判斷調(diào)整,是否需要測(cè)量,會(huì)繼續(xù)調(diào)用 onMeasure() 將 measureSpec 傳進(jìn)來(lái),測(cè)量尺寸的確定最終是在 onMeasure() 里完成的。

通常我們自定義 View 都要重寫這個(gè)方法實(shí)現(xiàn)自己的測(cè)量邏輯,包括我們常用的控件都是自己重寫了這個(gè)方法實(shí)現(xiàn)自己的測(cè)量邏輯。

如果不重寫 onMeasure(),會(huì)導(dǎo)致自定義 view 的 wrap_content 參數(shù)無(wú)效,具體可以看一下 getDefaultSize() 實(shí)現(xiàn)。

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
            getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));
}

public static int getDefaultSize(int size, int measureSpec) {
    int result = size;
    int specMode = MeasureSpec.getMode(measureSpec);
    int specSize = MeasureSpec.getSize(measureSpec);

    switch (specMode) {
    case MeasureSpec.UNSPECIFIED:
        result = size;
        break;
    case MeasureSpec.AT_MOST:
    case MeasureSpec.EXACTLY:
        //默認(rèn) 精確模式和最大模式下都是使用后邊的 specSize ,這會(huì)導(dǎo)致我們?cè)O(shè)置的 wrap_content 無(wú)效,始終是充滿父容器。
        result = specSize;
        break;
    }
    return result;
}

protected int getSuggestedMinimumHeight() {
    return (mBackground == null) ? mMinHeight : max(mMinHeight, mBackground.getMinimumHeight());

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

View 和 ViewGroup 的測(cè)量過(guò)程是不同的。

單純的 View 只需要在 onMeasure() 里完成自己的測(cè)量就可以了,ViewGroup 除了完成自己的測(cè)量外,還有子元素的測(cè)量。

ViewGroup 的 onMeasure() 是沒(méi)有任何實(shí)現(xiàn)的,因?yàn)楦鱾€(gè)布局的特性不同,具體測(cè)量邏輯也是不同的,具體實(shí)現(xiàn)都在各個(gè)布局里。

但是 ViewGroup 里提供了 measureChildren() 方法,思路就是,遍歷所有需要顯示的子元素,取出他們的 LayoutParams 參數(shù)在自己 measureSpec 的影響下創(chuàng)建出子元素的 measureSpec ,然后將調(diào)用子元素的 measure() 將measureSpec 傳遞進(jìn)去。

這里就將測(cè)量傳遞到了子元素。如果子元素是單純的 View 控件只需要完成自己就可以了,如果是 ViewGroup 會(huì)繼續(xù)將測(cè)量遞歸下去,直至完成整個(gè) View 樹(shù)的測(cè)量。

    protected void measureChildren(int widthMeasureSpec, int heightMeasureSpec) {
        final int size = mChildrenCount;
        final View[] children = mChildren;
        for (int i = 0; i < size; ++i) {
            final View child = children[i];
            if ((child.mViewFlags & VISIBILITY_MASK) != GONE) {
                //測(cè)量子元素,measureChild 見(jiàn)上面 MeasureSpec 里的代碼。
                measureChild(child, widthMeasureSpec, heightMeasureSpec);
            }
        }
    }

在完成測(cè)量流程之后就會(huì)進(jìn)入了 layout 流程了。

layout 布局流程

layout 這一流程會(huì)確定 View 的四個(gè)頂點(diǎn)位置,進(jìn)而確定在父容器中的位置和最終寬高。

layout 流程也是在 ViewRoot 里開(kāi)始,是在 performLayout() 里首先調(diào)用 DecorView 的 layout() 方法開(kāi)始整個(gè) View 樹(shù)的布局流程。

View 的布局流程都是在 layout() 方法里完成的,會(huì)在這里通過(guò) setFrame() 設(shè)置自己四個(gè)頂點(diǎn)的位置。

設(shè)置完自己的位置后,會(huì)繼續(xù)調(diào)用 onLayout() 方法,如果是 ViewGroup 可以繼續(xù)在 onLayout 里確定子元素的位置。

View 的 onLayout() 是沒(méi)有任何實(shí)現(xiàn)的,因?yàn)樗菦](méi)有子元素,ViewGroup 本身也是沒(méi)有實(shí)現(xiàn)的,也都是具體的各個(gè)布局里自己實(shí)現(xiàn)的。

思路也是遍歷所有需要布局的子元素,根據(jù)測(cè)量尺寸計(jì)算出他們的位置后調(diào)用子元素的 layout() 方法將位置參數(shù)穿進(jìn)去,讓子元素去完成自己的布局流程。

在這里也是將布局流程傳遞到了子元素,如果子元素是 ViewGroup 會(huì)繼續(xù)將布局流程傳遞,直到完成整個(gè) View 樹(shù)的布局流程。

  • layout() 確定自身的位置
  • onLayout() 確定子元素的位置

在完成 layout 流程后,就是最后一個(gè) draw 流程了。

draw 繪制流程

這個(gè)流程是將 View 繪制到屏幕上。

draw 流程也是在 ViewRoot 里開(kāi)始的,具體是在 performDraw() 里開(kāi)始,在這里會(huì)調(diào)用 DecorView 的 draw() 開(kāi)始整個(gè) View 樹(shù)的繪制。

draw 的過(guò)程相對(duì)來(lái)說(shuō)較為簡(jiǎn)單,在 draw() 里可以看到整個(gè)步驟

  1. 繪制背景 drawBackground(canvas);
  2. 繪制自己的內(nèi)容 onDraw(canvas);
  3. 繪制子元素 dispatchDraw(canvas);
  4. 繪制裝飾 onDrawForeground(canvas);

我們自定義 View 都會(huì)在 onDraw() 里實(shí)現(xiàn)自己的繪制邏輯,View 的 dispatchDraw() 是沒(méi)有任何實(shí)現(xiàn)的,具體實(shí)現(xiàn)在 ViewGroup 里。

在 ViewGroup 后調(diào)用子元素的 draw() 將繪制流程傳遞到子元素,直到繪制完整個(gè) View 樹(shù)。

在完成整個(gè) View 樹(shù)的繪制后,就可以在屏幕上看見(jiàn)界面了。

相關(guān)類 & 概念

在 View 的繪制過(guò)程中,涉及到了很多類,這里就不做詳細(xì)的介紹了,只在這里簡(jiǎn)單列一下,知道這些個(gè)的作用。

DecorView

整個(gè) View 樹(shù)的根節(jié)點(diǎn),所有的繪制,事件都是從這個(gè) View 開(kāi)始分發(fā)的。

它繼承自 FrameLayout 是一個(gè) ViewGroup ,內(nèi)部含有一個(gè) LinearLayout 。

這個(gè) LinearLayout 里有一個(gè) id 為 content 的 FrameLayout ,我們通常設(shè)置的 setContentView() 就是加載到了這個(gè) FrameLayout 里。

Window

每個(gè) Activity 都有一個(gè) window ,直譯就是“窗口”,是 Activity 的成員變量,也是應(yīng)用程序的視圖窗口,承載整個(gè) Activity 的視圖。 內(nèi)部含有一個(gè) DeocrView 成員變量,承載的視圖就是這個(gè) DeocrView 。

它目前只有一個(gè)實(shí)現(xiàn)類,PhoneWindow ,activity 里的 mWindow 就是這個(gè)實(shí)例。

ViewRoot

View Root 的作用很大,是連接 DecorView 和 Window Manager 的紐帶。 View 的繪制,觸屏,按鍵,屏幕刷新等事件分發(fā)都通過(guò)它完成的。

Activity 視圖結(jié)構(gòu)

以上就是Android view繪制流程詳解的詳細(xì)內(nèi)容,更多關(guān)于Android view繪制流程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Android如何給按鈕添加點(diǎn)擊音效

    Android如何給按鈕添加點(diǎn)擊音效

    這篇文章主要為大家詳細(xì)介紹了Android如何給按鈕添加點(diǎn)擊音效,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-03-03
  • Android實(shí)現(xiàn)3D標(biāo)簽云效果

    Android實(shí)現(xiàn)3D標(biāo)簽云效果

    這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)3D標(biāo)簽云效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-05-05
  • MobPush?Android常見(jiàn)問(wèn)題解決分析

    MobPush?Android常見(jiàn)問(wèn)題解決分析

    這篇文章主要介紹了MobPush?Android常見(jiàn)問(wèn)題解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-07-07
  • android 應(yīng)用退出時(shí)不播放動(dòng)畫的解決方法

    android 應(yīng)用退出時(shí)不播放動(dòng)畫的解決方法

    在Android應(yīng)用中,默認(rèn)情況下,當(dāng)用戶點(diǎn)擊返回按鈕退出應(yīng)用時(shí),系統(tǒng)會(huì)為應(yīng)用添加一個(gè)默認(rèn)的退出動(dòng)畫效果,本文將介紹如何在Android應(yīng)用中禁止退出動(dòng)畫的播放,感興趣的朋友一起看看吧
    2024-05-05
  • Android AnalogClock簡(jiǎn)單使用方法實(shí)例

    Android AnalogClock簡(jiǎn)單使用方法實(shí)例

    這篇文章主要介紹了Android AnalogClock簡(jiǎn)單使用方法,結(jié)合實(shí)例形式簡(jiǎn)單分析了AnalogClock的布局調(diào)用技巧,需要的朋友可以參考下
    2016-01-01
  • Android四大組件之broadcast廣播使用講解

    Android四大組件之broadcast廣播使用講解

    Android開(kāi)發(fā)的四大組件分別是:活動(dòng)(activity),用于表現(xiàn)功能;服務(wù)(service),后臺(tái)運(yùn)行服務(wù),不提供界面呈現(xiàn);廣播接受者(Broadcast Receive),勇于接收廣播;內(nèi)容提供者(Content Provider),支持多個(gè)應(yīng)用中存儲(chǔ)和讀取數(shù)據(jù),相當(dāng)于數(shù)據(jù)庫(kù),本篇著重介紹廣播組件
    2022-12-12
  • Android實(shí)現(xiàn)簡(jiǎn)易的音樂(lè)播放器

    Android實(shí)現(xiàn)簡(jiǎn)易的音樂(lè)播放器

    這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)簡(jiǎn)易的音樂(lè)播放器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-05-05
  • Android利用Intent啟動(dòng)和關(guān)閉Activity

    Android利用Intent啟動(dòng)和關(guān)閉Activity

    這篇文章主要為大家詳細(xì)介紹了Android利用Intent啟動(dòng)和關(guān)閉Activity的相關(guān)操作,感興趣的小伙伴們可以參考一下
    2016-06-06
  • Android多種方式實(shí)現(xiàn)相機(jī)圓形預(yù)覽的示例代碼

    Android多種方式實(shí)現(xiàn)相機(jī)圓形預(yù)覽的示例代碼

    這篇文章主要介紹了Android多種方式實(shí)現(xiàn)相機(jī)圓形預(yù)覽的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-08-08
  • android app icon 圖標(biāo)大小尺寸

    android app icon 圖標(biāo)大小尺寸

    應(yīng)用程序圖標(biāo) (Icon)應(yīng)當(dāng)是一個(gè) Alpha 通道透明的32位 PNG 圖片。由于安卓設(shè)備眾多,一個(gè)應(yīng)用程序圖標(biāo)需要設(shè)計(jì)幾種不同大小。
    2016-05-05

最新評(píng)論