【Android源碼】invalidate()

目錄:

@V2S5}}LDFY}OSO{MXDZWXV.png

在之前看書(shū)的時(shí)候,就知道了invalidate方法最終會(huì)去調(diào)用onDraw()方法,但是具體怎么調(diào)用卻一直沒(méi)深究過(guò)

我追尋源碼進(jìn)去呻顽,發(fā)現(xiàn)我們其實(shí)調(diào)用的是View中的invalidate()

ViewGroup.java

invalidate()

public void invalidate(boolean invalidateCache) {
    invalidateInternal(0, 0, mRight - mLeft, mBottom - mTop, invalidateCache, true);
}

我們繼續(xù)跟蹤下去雹顺,

invalidateInternal()

void invalidateInternal(int l, int t, int r, int b, boolean invalidateCache,
        boolean fullInvalidate) {
    //...
    final ViewParent p = mParent;
    if (p != null && ai != null && l < r && t < b) {
        final Rect damage = ai.mTmpInvalRect;
        damage.set(l, t, r, b);
        p.invalidateChild(this, damage);
    }
    //...
}

這里可以看到,我們mParent對(duì)象調(diào)用了invalidateChild()方法廊遍,

invalidateChild()

public final void invalidateChild(View child, final Rect dirty) {
    //...
    do {
        parent = parent.invalidateChildInParent(location, dirty);
    } while (parent != null);
    //...
}

這個(gè)方法里面嬉愧,do~while循環(huán)中一直在循環(huán)調(diào)用invalidateChildInParent()方法,通過(guò)這個(gè)操作喉前,可以一直向上查找父布局没酣,直到找到最頂層的View,當(dāng)他沒(méi)有parentView時(shí)卵迂,就會(huì)退出這個(gè)循環(huán)

invalidateChildInParent()

public ViewParent invalidateChildInParent(final int[] location, final Rect dirty) {
    //...
    return mParent;
}

在內(nèi)部的這些View都是調(diào)用的是ViewGroup的invalidateChildInParent()方法裕便,但是當(dāng)我們向上查找到最外層時(shí),最外層View是沒(méi)有父布局的见咒,所以這個(gè)時(shí)候闪金,最外層View會(huì)調(diào)用View中的invalidateChildInParent()方法。我們現(xiàn)在就去找最外層的View,在ViewRootImpl類中

ViewRootImpl.java

invalidateChildInParent()

public ViewParent invalidateChildInParent(int[] location, Rect dirty) {
    checkThread();
    //...
    invalidateRectOnScreen(dirty);
}

追到這里時(shí),我們發(fā)現(xiàn)了另一個(gè)知識(shí)點(diǎn)角虫,==為什么不能在子線程中更新UI?==

//TextView.java
private void setText(CharSequence text, BufferType type,
                         boolean notifyBefore, int oldlen) {
    //...
    if (mLayout != null) {
        checkForRelayout();
    }
    //...
}
private void checkForRelayout() {
    if (//...){
        //...
        invalidate();
    } else {
        //...
        invalidate();
    }
}


//ImageView.java
public void setImageResource(@DrawableRes int resId) {
    //...
    invalidate();
}

因?yàn)槲覀兏耈I時(shí)漏设,如setText()、setImageResource()等等今妄,都會(huì)先去調(diào)用invalidate()方法郑口,然后最終去調(diào)用checkThread()這個(gè)方法,而這個(gè)方法就是用來(lái)檢測(cè)線程

checkThread()

void checkThread() {
    if (mThread != Thread.currentThread()) {
        throw new CalledFromWrongThreadException(
                "Only the original thread that created a view hierarchy can touch its views.");
    }
}

上面的if判斷中盾鳞,mThread是在ViewRootImpl的構(gòu)造函數(shù)中創(chuàng)建的犬性,而Thread.currentThread()是返回的當(dāng)前線程,所以如果是在在一個(gè)線程中(如主線程)開(kāi)的子線程腾仅,那么Thread.currentThread()就代表的子線程乒裆,而mThread就代表的原始線程,所以就會(huì)拋出異常

好推励,現(xiàn)在回到ViewRootImpl的invalidateChildInParent()方法中鹤耍,除了checkThread()方法外,invalidateRectOnScreen()方法是值得我們關(guān)注的验辞,

invalidateRectOnScreen()

private void invalidateRectOnScreen(Rect dirty) {
    //...
    if (!mWillDrawSoon && (intersected || mIsAnimating)) {
        scheduleTraversals();
    }
}

在他里面稿黄,我們看到有scheduleTraversals()方法,繼續(xù)追

scheduleTraversals()

void scheduleTraversals() {
    if (!mTraversalScheduled) {
       //...
        mChoreographer.postCallback(
            Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
        //...
    }
}

到了這里跌造,我們看mTraversalRunnable這個(gè)對(duì)象杆怕,

final TraversalRunnable mTraversalRunnable = new TraversalRunnable();
final class TraversalRunnable implements Runnable {
    @Override
    public void run() {
        doTraversal();
    }
}

doTraversal()

void doTraversal() {
    if (mTraversalScheduled) {
        //...
        performTraversals();
        //...
    }
}

經(jīng)過(guò)一些列的追蹤,我們到了這里,我們找到了performTraversals()這個(gè)方法陵珍,這個(gè)方法在UI繪制里面是非常重要的寝杖,我們看一下這個(gè)方法的方法體

performTraversals()

private void performTraversals() {
    //...
    performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
    //...
    performLayout(lp, mWidth, mHeight);
    //...
    performDraw();
    //...
}

它內(nèi)部的這三個(gè)方法正是我們UI繪制流程中的三個(gè)流程:measure、layout撑教、draw朝墩,所以說(shuō)他是UI繪制流程中非常重要的方法

我們這里主要看一下performDraw()方法

performDraw()

private void performDraw() {
    //...
    boolean canUseAsync = draw(fullRedrawNeeded);
    //...
}

它里面有一個(gè)draw()方法

draw()

private boolean draw(boolean fullRedrawNeeded) {
    //...
    if (!drawSoftware(surface, mAttachInfo, xOffset, yOffset,
                        scalingRequired, dirty, surfaceInsets)) {
        return false;
    }
    //...
}

可以看到醉拓,這里有一個(gè)drawSoftware()方法

drawSoftware()

private boolean drawSoftware(Surface surface, AttachInfo attachInfo, int xoff, int yoff,
        boolean scalingRequired, Rect dirty, Rect surfaceInsets) {
    // Draw with software renderer.
    final Canvas canvas;
    //...
    canvas = mSurface.lockCanvas(dirty);
    //...
    mView.draw(canvas);
    //...
}

到這里伟姐,我們終于看到了Canvas對(duì)象,它的實(shí)現(xiàn)亿卤,也是在surface上面實(shí)現(xiàn)的愤兵,并且他所展示的區(qū)域也是我們?cè)谕饷鎮(zhèn)鬟M(jìn)來(lái)的dirty

最后,我們看到了mView.draw(canvas)這里的mView不是我們最開(kāi)始向上查找父布局的View排吴,而是最外層的View秆乳,即沒(méi)有父布局的View

draw()

public void draw(Canvas canvas) {
    /*
     * Draw traversal performs several drawing steps which must be executed
     * in the appropriate order:
     *
     *      1. Draw the background
     *      2. If necessary, save the canvas' layers to prepare for fading
     *      3. Draw view's content
     *      4. Draw children
     *      5. If necessary, draw the fading edges and restore layers
     *      6. Draw decorations (scrollbars for instance)
     *      7. If necessary, draw the default focus highlight
     */
}

它內(nèi)部的注釋說(shuō)得很清楚,它先會(huì)去繪制background钻哩,然后繪制自己的內(nèi)容屹堰,最后繪制自己的子View,就這么個(gè)流程

總結(jié)

最后的最后街氢,我們?cè)賮?lái)梳理一下invlidate()的繪制流程:

我們先找到當(dāng)前的View扯键,然后它通過(guò)一個(gè)do~While循環(huán)一直向上查找最外層View,當(dāng)我們找到最外層View后珊肃,會(huì)調(diào)用最外層View的Draw方法荣刑,而draw方法流程中的dispatchDraw(canvas)就會(huì)一直往下畫,最終畫到最開(kāi)始調(diào)用invlidate()的當(dāng)前的View

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末伦乔,一起剝皮案震驚了整個(gè)濱河市厉亏,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌烈和,老刑警劉巖爱只,帶你破解...
    沈念sama閱讀 222,729評(píng)論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異招刹,居然都是意外死亡虱颗,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,226評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門蔗喂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)忘渔,“玉大人,你說(shuō)我怎么就攤上這事缰儿∑枇福” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 169,461評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)宣赔。 經(jīng)常有香客問(wèn)我预麸,道長(zhǎng),這世上最難降的妖魔是什么儒将? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 60,135評(píng)論 1 300
  • 正文 為了忘掉前任吏祸,我火速辦了婚禮,結(jié)果婚禮上钩蚊,老公的妹妹穿的比我還像新娘贡翘。我一直安慰自己,他們只是感情好砰逻,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,130評(píng)論 6 398
  • 文/花漫 我一把揭開(kāi)白布鸣驱。 她就那樣靜靜地躺著,像睡著了一般蝠咆。 火紅的嫁衣襯著肌膚如雪踊东。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 52,736評(píng)論 1 312
  • 那天刚操,我揣著相機(jī)與錄音闸翅,去河邊找鬼。 笑死菊霜,一個(gè)胖子當(dāng)著我的面吹牛坚冀,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播占卧,決...
    沈念sama閱讀 41,179評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼遗菠,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了华蜒?” 一聲冷哼從身側(cè)響起辙纬,我...
    開(kāi)封第一講書(shū)人閱讀 40,124評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎叭喜,沒(méi)想到半個(gè)月后贺拣,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,657評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡捂蕴,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,723評(píng)論 3 342
  • 正文 我和宋清朗相戀三年譬涡,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片啥辨。...
    茶點(diǎn)故事閱讀 40,872評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡涡匀,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出溉知,到底是詐尸還是另有隱情陨瘩,我是刑警寧澤腕够,帶...
    沈念sama閱讀 36,533評(píng)論 5 351
  • 正文 年R本政府宣布,位于F島的核電站舌劳,受9級(jí)特大地震影響帚湘,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜甚淡,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,213評(píng)論 3 336
  • 文/蒙蒙 一大诸、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧贯卦,春花似錦资柔、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,700評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)盈厘。三九已至睁枕,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間沸手,已是汗流浹背外遇。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,819評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留契吉,地道東北人跳仿。 一個(gè)月前我還...
    沈念sama閱讀 49,304評(píng)論 3 379
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像捐晶,于是被迫代替她去往敵國(guó)和親菲语。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,876評(píng)論 2 361

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