Android @ View 的繪制流程

  1. measure毅人、layout、draw
  2. 繪制的入口是由 ViewRootImpl 的 performTraversals 方法來(lái)發(fā)起measure,layout谎替,draw等流程的
  3. 父 View 的 measure 的過(guò)程會(huì)先測(cè)量子 View,等子 View 測(cè)量結(jié)果出來(lái)后蹋辅,再來(lái)測(cè)量自己
  4. MeasureSpec(AT_MOST钱贯、EXACTLY、UNSPECIFIED)由父 View 的
    MeasureSpec 和子 View 的 LayoutParams 通過(guò)簡(jiǎn)單的計(jì)算得出一個(gè)針對(duì)子 View 的測(cè)量要求
  5. measure 是 final 的侦另,所以自定義 View 重寫(xiě) onMeasure

ViewRootImpl 中的相關(guān)源碼:

private void performTraversals() {
    ......
    int childWidthMeasureSpec = getRootMeasureSpec(mWidth, lp.width);
    int childHeightMeasureSpec = getRootMeasureSpec(mHeight, lp.height);
    ......
    mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);
    ......
    mView.layout(0, 0, mView.getMeasuredWidth(), mView.getMeasuredHeight());
    ......
    mView.draw(canvas);
    ......
}

private static int getRootMeasureSpec(int windowSize, int rootDimension) {
    int measureSpec;
    switch (rootDimension) {
    case ViewGroup.LayoutParams.MATCH_PARENT:
        measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.EXACTLY);
        break;
    case ViewGroup.LayoutParams.WRAP_CONTENT:
        measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.AT_MOST);
        break;
    default:
        measureSpec = MeasureSpec.makeMeasureSpec(rootDimension, MeasureSpec.EXACTLY);
        break;
    }
    return measureSpec;
}

FrameLayout 中的相關(guān)源碼:

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    ......
    int maxHeight = 0;
    int maxWidth = 0;
    int childState = 0;
    for (int i = 0; i < count; i++) {
        final View child = getChildAt(i);
        if (mMeasureAllChildren || child.getVisibility() != GONE) {
            measureChildWithMargins(child, widthMeasureSpec, 0, heightMeasureSpec, 0);
            final LayoutParams lp = (LayoutParams) child.getLayoutParams();
            maxWidth = Math.max(maxWidth, child.getMeasuredWidth() + lp.leftMargin + lp.rightMargin);
            maxHeight = Math.max(maxHeight, child.getMeasuredHeight() + lp.topMargin + lp.bottomMargin);
            ......
        }
    }
    ......
    setMeasuredDimension(resolveSizeAndState(maxWidth, widthMeasureSpec, childState),
            resolveSizeAndState(maxHeight, heightMeasureSpec, childState << MEASURED_HEIGHT_STATE_SHIFT));
    ......
}

ViewGroup 中的相關(guān)源碼:

protected void measureChildWithMargins(View child, int parentWidthMeasureSpec, int widthUsed, int parentHeightMeasureSpec, int heightUsed) {
    final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
    final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
            mPaddingLeft + mPaddingRight + lp.leftMargin + lp.rightMargin + widthUsed, lp.width);
    final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,
            mPaddingTop + mPaddingBottom + lp.topMargin + lp.bottomMargin + heightUsed, lp.height);
    child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}

public static int getChildMeasureSpec(int spec, int padding, int childDimension) {
    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) {
        case MeasureSpec.EXACTLY:
            if (childDimension >= 0) {
                resultSize = childDimension;
                resultMode = MeasureSpec.EXACTLY;
            } else if (childDimension == LayoutParams.MATCH_PARENT) {
                resultSize = size;
                resultMode = MeasureSpec.EXACTLY;
            } else if (childDimension == LayoutParams.WRAP_CONTENT) {
                resultSize = size;
                resultMode = MeasureSpec.AT_MOST;
            }
            break;
        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;
        case MeasureSpec.UNSPECIFIED:
            if (childDimension >= 0) {
                resultSize = childDimension;
                resultMode = MeasureSpec.EXACTLY;
            } else if (childDimension == LayoutParams.MATCH_PARENT) {
                resultSize = 0;
                resultMode = MeasureSpec.UNSPECIFIED;
            } else if (childDimension == LayoutParams.WRAP_CONTENT) {
                resultSize = 0;
                resultMode = MeasureSpec.UNSPECIFIED;
            }
            break;
    }
    return MeasureSpec.makeMeasureSpec(resultSize, resultMode);
}

View 中的相關(guān)源碼:

public final void measure(int widthMeasureSpec, int heightMeasureSpec) {
    ......
    onMeasure(widthMeasureSpec, heightMeasureSpec);
    ......
}

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    // 給 mMeasuredWidth 和 mMeasuredHeight 設(shè)值秩命,如果這兩個(gè)值一旦設(shè)置了,那么意味著對(duì)于這個(gè)View的測(cè)量結(jié)束了褒傅,這個(gè) View 的寬高已經(jīng)有測(cè)量的結(jié)果出來(lái)了弃锐。
    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:
        result = specSize;
        break;
    }
    return result;
}

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

參考:

  1. Android View的繪制流程

未完成...

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市殿托,隨后出現(xiàn)的幾起案子霹菊,更是在濱河造成了極大的恐慌,老刑警劉巖支竹,帶你破解...
    沈念sama閱讀 222,865評(píng)論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件旋廷,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡礼搁,警方通過(guò)查閱死者的電腦和手機(jī)饶碘,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,296評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)馒吴,“玉大人扎运,你說(shuō)我怎么就攤上這事卑雁。” “怎么了绪囱?”我有些...
    開(kāi)封第一講書(shū)人閱讀 169,631評(píng)論 0 364
  • 文/不壞的土叔 我叫張陵测蹲,是天一觀(guān)的道長(zhǎng)。 經(jīng)常有香客問(wèn)我鬼吵,道長(zhǎng)扣甲,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 60,199評(píng)論 1 300
  • 正文 為了忘掉前任齿椅,我火速辦了婚禮琉挖,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘涣脚。我一直安慰自己示辈,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,196評(píng)論 6 398
  • 文/花漫 我一把揭開(kāi)白布遣蚀。 她就那樣靜靜地躺著矾麻,像睡著了一般。 火紅的嫁衣襯著肌膚如雪芭梯。 梳的紋絲不亂的頭發(fā)上险耀,一...
    開(kāi)封第一講書(shū)人閱讀 52,793評(píng)論 1 314
  • 那天,我揣著相機(jī)與錄音玖喘,去河邊找鬼甩牺。 笑死,一個(gè)胖子當(dāng)著我的面吹牛累奈,可吹牛的內(nèi)容都是我干的贬派。 我是一名探鬼主播,決...
    沈念sama閱讀 41,221評(píng)論 3 423
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼澎媒,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼搞乏!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起旱幼,我...
    開(kāi)封第一講書(shū)人閱讀 40,174評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤查描,失蹤者是張志新(化名)和其女友劉穎突委,沒(méi)想到半個(gè)月后柏卤,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,699評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡匀油,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,770評(píng)論 3 343
  • 正文 我和宋清朗相戀三年缘缚,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片敌蚜。...
    茶點(diǎn)故事閱讀 40,918評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡桥滨,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情齐媒,我是刑警寧澤蒲每,帶...
    沈念sama閱讀 36,573評(píng)論 5 351
  • 正文 年R本政府宣布,位于F島的核電站喻括,受9級(jí)特大地震影響邀杏,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜唬血,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,255評(píng)論 3 336
  • 文/蒙蒙 一望蜡、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧拷恨,春花似錦脖律、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,749評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至冕杠,卻和暖如春膏孟,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背拌汇。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,862評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工柒桑, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人噪舀。 一個(gè)月前我還...
    沈念sama閱讀 49,364評(píng)論 3 379
  • 正文 我出身青樓拯辙,卻偏偏與公主長(zhǎng)得像蛋济,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,926評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容

  • View的繪制和事件處理是兩個(gè)重要的主題楷怒,上一篇《圖解 Android事件分發(fā)機(jī)制》已經(jīng)把事件的分發(fā)機(jī)制講得比較詳...
    Kelin閱讀 119,732評(píng)論 100 845
  • 當(dāng)Android原生控件無(wú)法滿(mǎn)足需求時(shí)就要自定義View净响,只有掌握了View的測(cè)量過(guò)程 (measure)少欺、布局過(guò)...
    小蕓論閱讀 7,277評(píng)論 7 54
  • Android的UI管理系統(tǒng)層級(jí)關(guān)系 PhoneWindow是Adroid系統(tǒng)中最基本的窗口系統(tǒng),每個(gè)Activi...
    凱玲之戀閱讀 1,888評(píng)論 0 2
  • 概述 本篇文章會(huì)從源碼(基于A(yíng)ndroid 6.0)角度分析Android中View的繪制流程馋贤,側(cè)重于對(duì)整體流程的...
    absfree閱讀 76,972評(píng)論 24 273
  • 那年你叛逆的從母親的懷里脫落 你說(shuō)想去遠(yuǎn)方赞别,跟隨著小溪 路上遇見(jiàn)了流浪的詩(shī)人, 他用詩(shī)句贊頌著野花配乓,叢林仿滔,還有載你...
    丿肆悅閱讀 290評(píng)論 6 9