国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

ViewRootImpl如何負責管理繪制視圖樹和刷新界面

開發 前端
mHandler是當前主線程的handler,當接收到onVsync信號的時候,將自己封裝到Message中,等到Looper處理,最后Looper處理消息的時候就會調用run方法最終從mCallbackQueues取回之前添加的任務再執行run方法,也就是TraservalRunnable的run方法。

ViewRootImpl位于視圖層次結構的頂部,負責View和WindowManager之間的通信。

  1. 視圖繪制:負責調用View樹的繪制流程,包括測量(measure)、布局(layout)和繪制(draw)等操作。監聽View樹的改變,并根據需要觸發相應的繪制操作,確保界面的及時更新。
  2. 事件分發:負責將輸入的各種事件(如觸摸事件、按鍵事件等)分發給正確的View處理。根據觸摸事件的位置信息,逐層遍歷View樹,找到最合適的View來處理事件。將處理結果返回給系統,以便進行后續的處理,如滾動、點擊等。
  3. 窗口管理:承擔Android窗口管理的一部分職責。負責創建和管理窗口,將窗口與ViewRootImpl進行綁定。當窗口需要顯示或隱藏時,ViewRootImpl會相應地調整界面的顯示狀態。

ViewRootImpl關聯了多個類和接口,IWindowSession、Choreographer及其FrameCallback接口等。IWindowSession用于客戶端和WindowManagerService之間進行窗口管理操作的接口,允許ViewRootImpl與WindowManagerService進行通信,執行如添加、刪除、更新窗口等操作。

scheduleTraversals

scheduleTraversals()方法負責將一次視圖遍歷(traversal)排期到其調度計劃中,但并不會立即執行遍歷操作。方法被許多操作所調用,比如當視圖的大小、位置等屬性發生變化時,或者當調用requestLayout()、invalidate()等方法時,都會觸發scheduleTraversals()。作用是將視圖的測量、布局和繪制操作(即遍歷操作)放入待執行隊列中,并注冊一個底層的刷新信號監聽器。

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

void invalidateInternal(int l, int t, int r, int b, boolean invalidateCache, boolean fullInvalidate) {
    ...
    // Propagate the damage rectangle to the parent view.
    final AttachInfo ai = mAttachInfo;
    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);
    }
    ...
}

public final void invalidateChild(View child, final Rect dirty) {
    .....
    ViewParent parent = this;
        do {
            View view = null;
            if (parent instanceof View) {
                view = (View) parent;
            }

            if (drawAnimation) {
                if (view != null) {
                    view.mPrivateFlags |= PFLAG_DRAW_ANIMATION;
                } else if (parent instanceof ViewRootImpl) {
                    ((ViewRootImpl) parent).mIsAnimating = true;
                }
            }

            // If the parent is dirty opaque or not dirty, mark it dirty with the opaque
            // flag coming from the child that initiated the invalidate
            if (view != null) {
                if ((view.mViewFlags & FADING_EDGE_MASK) != 0 && view.getSolidColor() == 0) {
                    opaqueFlag = PFLAG_DIRTY;
                }
                if ((view.mPrivateFlags & PFLAG_DIRTY_MASK) != PFLAG_DIRTY) {
                    view.mPrivateFlags = (view.mPrivateFlags & ~PFLAG_DIRTY_MASK) | opaqueFlag;
                }
            }
            //調用ViewGrup的invalidateChildInParent,如果已經達到最頂層view,則調用ViewRootImpl的invalidateChildInParent。
            parent = parent.invalidateChildInParent(location, dirty);
            if (view != null) {
                // Account for transform on current parent
                Matrix m = view.getMatrix();
                if (!m.isIdentity()) {
                    RectF boundingRect = attachInfo.mTmpTransformRect;
                    boundingRect.set(dirty);
                    m.mapRect(boundingRect);
                    dirty.set((int) Math.floor(boundingRect.left),
                              (int) Math.floor(boundingRect.top),
                              (int) Math.ceil(boundingRect.right),
                              (int) Math.ceil(boundingRect.bottom));
                }
            }
        } while (parent != null);
    }
}

當VSYNC信號到來時(VSYNC信號是Android系統中用于同步屏幕刷新的信號),系統會從待執行隊列中取出對應的scheduleTraversals()操作,并將其加入到主線程的消息隊列中。然后,主線程會從消息隊列中取出并執行這個操作,進而觸發視圖的測量、布局和繪制流程。

private void scheduleFrameLocked(long now) {
    if (!mFrameScheduled) {
        mFrameScheduled = true;
        if (USE_VSYNC) {
            //這里判斷,當前執行的線程是否是創建該Choreographer的線程,如果是直接執行。否則通過handler 發送到 創建該Choreographer的線程去執行。
            if (isRunningOnLooperThreadLocked()) {
                scheduleVsyncLocked();
            } else {
                //這條message 最后處理還是調用到了scheduleVsyncLocked方法
                Message msg = mHandler.obtainMessage(MSG_DO_SCHEDULE_VSYNC);
                msg.setAsynchronous(true);
                mHandler.sendMessageAtFrontOfQueue(msg);
            }
        } else {
            final long nextFrameTime = Math.max(
                    mLastFrameTimeNanos / TimeUtils.NANOS_PER_MS + sFrameDelay, now);
            if (DEBUG_FRAMES) {
                Log.d(TAG, "Scheduling next frame in " + (nextFrameTime - now) + " ms.");
            }
            Message msg = mHandler.obtainMessage(MSG_DO_FRAME);
            msg.setAsynchronous(true);
            mHandler.sendMessageAtTime(msg, nextFrameTime);
        }
    }
}

private void scheduleVsyncLocked() {
    mDisplayEventReceiver.scheduleVsync();
}

public void scheduleVsync() {
    if (mReceiverPtr == 0) {
        Log.w(TAG, "Attempted to schedule a vertical sync pulse but the display event " + "receiver has already been disposed.");
    } else {
        nativeScheduleVsync(mReceiverPtr);
    }
}

在這個過程中,performTraversals()方法會被調用。方法會執行實際的測量、布局和繪制操作。首先會調用measureHierarchy()方法進行測量,然后調用performLayout()方法進行布局,最后調用draw()方法進行繪制。這些操作會按照順序執行,以確保視圖能夠正確地顯示在屏幕上。最終通過nativeScheduleVsync()原生方法通知屏幕進行繪制。

performTraversals

performTraversals()方法負責啟動視圖的測量(measure)、布局(layout)和繪制(draw)流程。當需要創建視圖、視圖參數改變或界面需要刷新時,可能會從根視圖DecorView開始重新進行測量、布局和繪制,這時就會調用到performTraversals()方法。

private void performTraversals() {
    ...
    performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
    ...
    performLayout(lp, desiredWindowWidth, desiredWindowHeight);
    ... 
    performDraw();
}

void doTraversal() {
      //防止重入
    if (mTraversalScheduled) {
        mTraversalScheduled = false;
        //移除同步屏障
        mHandler.getLooper().getQueue()
                    .removeSyncBarrier(mTraversalBarrier);
        performTraversals();
    }
}

圖片圖片

  1. 「測量(Measure)」:在這個階段,系統會遍歷整個視圖樹,計算每個視圖的大小。這個過程中會用到MeasureSpec,每個MeasureSpec都包含了一個測量模式和測量大小,測量模式主要有三種:EXACTLY(父視圖已經確定了子視圖的確切大小)、AT_MOST(子視圖的大小有一個最大值限制)和UNSPECIFIED(父視圖對子視圖的大小沒有要求)。
  2. 「布局(Layout)」:在測量完成后,系統會為每個視圖確定其在屏幕上的精確位置。這個過程中,父視圖會根據測量階段得到的子視圖大小以及自身的布局參數,計算出子視圖應該放置的位置。
  3. 「繪制(Draw)」:系統會遍歷整個視圖樹,根據每個視圖的繪制參數(如顏色、形狀等)將其繪制到屏幕上。這個過程中,視圖會按照其在視圖樹中的層次順序進行繪制,先繪制父視圖,再繪制子視圖。
final TraversalRunnable mTraversalRunnable = new TraversalRunnable();

void scheduleTraversals() {
    if (!mTraversalScheduled) {
        //移除同步屏障
        mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();
        mChoreographer.postCallback(
                Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
           
    }
}

void unscheduleTraversals() {
    mChoreographer.removeCallbacks(
                    Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
    }
}

public void postCallback(int callbackType, Runnable action, Object token) {
    postCallbackDelayed(callbackType, action, token, 0);
}

public void postCallbackDelayed(int callbackType,
    Runnable action, Object token, long delayMillis) {
    ...
    postCallbackDelayedInternal(callbackType, action, token, delayMillis);
}

private void postCallbackDelayedInternal(int callbackType,
    Object action, Object token, long delayMillis) {
    ....
    synchronized (mLock) {
        final long now = SystemClock.uptimeMillis();
        final long dueTime = now + delayMillis;
        //把 任務添加到了mCallbackQueues 回調里面去,等待回調執行。
        mCallbackQueues[callbackType].addCallbackLocked(dueTime, action, token);
            
        //now=0 ,走進scheduleFrameLocked()方法內
        if (dueTime <= now) {
            scheduleFrameLocked(now);
        } else {
            Message msg = mHandler.obtainMessage(MSG_DO_SCHEDULE_CALLBACK, action);
            msg.arg1 = callbackType;
            msg.setAsynchronous(true);
            mHandler.sendMessageAtTime(msg, dueTime);
        }
    }
}

//Choreographer內部類DisplayEventReceiver,重寫了onVsync方法
@Override
public void onVsync(long timestampNanos, int builtInDisplayId, int frame) {
    mTimestampNanos = timestampNanos;
    mFrame = frame;
    Message msg = Message.obtain(mHandler, this);
    // 設置成異步消息
    msg.setAsynchronous(true);
    mHandler.sendMessageAtTime(msg, timestampNanos / TimeUtils.NANOS_PER_MS);
}

public void run() {
    mHavePendingVsync = false;
    doFrame(mTimestampNanos, mFrame);
}

// Choreographer
void doFrame(long frameTimeNanos, int frame) {
    ...
    doCallbacks(Choreographer.CALLBACK_TRAVERSAL, frameTimeNanos);
}

void doCallbacks(int callbackType, long frameTimeNanos) {
    CallbackRecord callbacks;
    // 從mCallbackQueues取出
    callbacks = mCallbackQueues[callbackType].extractDueCallbacksLocked(now / TimeUtils.NANOS_PER_MS);
    for (CallbackRecord c = callbacks; c != null; c = c.next) {
         c.run(frameTimeNanos);
    }
}
// CallbackRecord
public void run(long frameTimeNanos) {
    if (token == FRAME_CALLBACK_TOKEN) {
        ((FrameCallback)action).doFrame(frameTimeNanos);
    } else {
        // 這里也即是調用了TraservalRunnable的run方法,也即是三個繪制流程
        ((Runnable)action).run();
    }
}

mHandler是當前主線程的handler,當接收到onVsync信號的時候,將自己封裝到Message中,等到Looper處理,最后Looper處理消息的時候就會調用run方法最終從mCallbackQueues取回之前添加的任務再執行run方法,也就是TraservalRunnable的run方法。最終觸發performTraversals方法進行界面刷新。

責任編輯:武曉燕 來源: 沐雨花飛蝶
相關推薦

2009-12-25 18:06:11

WPF刷新界面

2009-08-31 09:13:00

UbuntuNetBook Rem界面

2025-01-07 15:23:47

iOS 18iOS 19蘋果

2009-07-10 08:50:35

微軟Windows 7界面

2012-07-17 09:53:02

2012-06-18 10:57:25

Windows 8操作系統

2010-04-15 09:47:02

2011-12-29 10:13:48

FirefoxAndroid版

2017-05-12 09:29:42

操作系統Windows 10 win 10 NEON

2010-08-05 09:17:17

MeeGo界面

2011-04-14 13:30:55

webOS 3.0webOS惠普

2009-12-23 20:45:09

Firefox全新界面

2012-05-11 16:11:50

Visual Stud

2022-07-06 09:54:46

微軟Windows 11

2023-05-04 06:31:45

OneDrive微軟

2009-02-12 15:18:59

2023-05-04 09:44:17

開源FydeOS

2011-08-18 09:30:39

金山Kingsoft Of

2012-03-26 11:00:10

Visual Stud微軟開發

2012-08-15 09:26:30

SkyDriveOutlook.com
點贊
收藏

51CTO技術棧公眾號

日本激情视频网| 黄色免费在线播放| 日韩午夜高潮| 欧美一性一乱一交一视频| sm久久捆绑调教精品一区| 精品久久香蕉国产线看观看亚洲| 福利在线一区二区三区| 99这里都是精品| 人妻激情另类乱人伦人妻| 日韩精品一二三区| 国产伦精品一区二区三| 国产一区视频在线观看免费| 99理论电影网| 激情综合中文娱乐网| 国产精品三区在线| 在线亚洲成人| 五月天亚洲综合| 精品一区二区三区视频在线观看 | 中国色在线观看另类| 欧美激情国产精品日韩| 国产色产综合产在线视频| 国产亚洲精品网站| 中文字幕欧美三区| 免费三级欧美电影| 精品av在线播放| www日韩tube| 精品国产一区二区三区久久久蜜月| 久草在线视频福利| 尤物精品国产第一福利三区| 四虎影视成人精品国库在线观看| 久久精品国产一区| 欧美xxxx在线| 国产欧美精品日韩精品| 99视频精品全国免费| 91在线免费看片| 久久xxxx精品视频| 欧洲精品在线播放| 欧美激情资源网| 亚洲日本va中文字幕久久| 337p亚洲精品色噜噜狠狠| 日韩伦理在线一区| 久久久人成影片一区二区三区观看| 在线看成人短视频| 国产伦精品一区二区三区照片| 久久久夜精品| 国产精品一二三四区| 天堂8在线天堂资源bt| 久久蜜桃av一区二区天堂| 久草在线在线视频| 在线免费亚洲电影| www.com.cn成人| 欧美在线免费看| 一区二区亚洲精品| 久久人人爽人人爽人人av| 亚洲视频一区二区在线观看| 国产玉足榨精视频在线观看| 亚洲精品美女久久久久| 久久久久观看| 久久久影院一区二区三区| 国产jizzjizz一区二区| 日本中文字幕电影| 亚洲精品一区二区三区香蕉| 欧美电影免费网站| 欧美在线激情| 国产精品久久久久影院| 岛国中文字幕在线| 97视频免费观看| 午夜亚洲激情| 三上悠亚在线资源| 精品久久久久99| 午夜视频免费在线观看| 欧美一区二区三区四区视频| 国产一区二区三区免费观看在线| 97人人干人人| 91在线国产福利| 国产中文字幕在线观看| 日韩在线视频导航| 亚洲高清在线| 成人动漫h在线观看| 日韩亚洲欧美在线| 凹凸成人精品亚洲精品密奴| 嫩草影院中文字幕| 欧洲亚洲国产日韩| 精品视频一区二区三区在线观看| 欧美激情第六页| 午夜欧美一区二区三区在线播放| 一区二区三区日本视频| 色大师av一区二区三区| 亚洲成va人在线观看| 91精品国产一区二区在线观看 | 69久久99精品久久久久婷婷| 精品少妇3p| 老司机午夜网站| 欧美电影一区二区| 成人在线一区| 99热手机在线| 亚洲精选一区二区| 亚洲久久视频| 在线看片免费人成视久网| 久久国产天堂福利天堂| 狠狠色丁香久久婷婷综| 免费成人黄色| 3d动漫精品啪啪一区二区三区免费 | 99国产精品国产精品毛片| 国产美女视频一区二区三区| 琪琪第一精品导航| 91老师国产黑色丝袜在线| 在线观看特色大片免费视频| 欧美日韩国产精品一区二区| 欧美综合天天夜夜久久| 国产精品97| 在线午夜视频| 欧亚精品中文字幕| 中文字幕一区二区不卡| 中文字幕久久精品一区二区| 久久精品国产sm调教网站演员| 亚洲久久久久久久久久| 美日韩一区二区| free性欧美16hd| 一区二区免费在线观看| 日韩欧美美女一区二区三区| 国产日韩欧美一区在线| 日本福利专区在线观看| 久久精品国产美女| 欧美一区二区视频在线观看2022| 亚洲激情偷拍| 欧美69xxx| 日韩影视精品| 亚洲国产精品高清久久久| 日本在线观看不卡视频| jizz性欧美| 日韩在线导航| 亚洲精品国产精品久久清纯直播| 蜜桃av噜噜一区| 中文字幕在线看片| 麻豆tv在线播放| 欧美老肥婆性猛交视频| 中文字幕免费一区| 香蕉久久夜色精品国产使用方法| www亚洲天堂| 国产精品自产拍在线观看中文| 午夜日韩在线观看| 欧美成人嫩草网站| 欧美成人性生活视频| 美女黄毛**国产精品啪啪| 精品日韩一区二区三区免费视频| 老司机免费视频一区二区 | 欧美日韩高清一区二区三区| 一区二区动漫| 国产精选在线| 国产男女免费视频| 91精品国产91久久久久| 欧美日韩视频在线| 久久九九电影| 99国内精品久久久久| 69日本xxxxxxxxx49| 999精品在线观看| 亚洲缚视频在线观看| 白白色 亚洲乱淫| 欧美挤奶吃奶水xxxxx| 日韩av免费观影| 夜夜春亚洲嫩草影视日日摸夜夜添夜| 在线成人激情视频| 亚洲免费观看高清完整版在线观看| 一个色综合网| 亚洲精品永久免费视频| 成人看片app| 久久国产精品-国产精品| 在线观看国产精品淫| 亚洲乱码国产乱码精品精可以看| 一区在线免费| 免费在线成人激情电影| 国产成免费视频| 国产精品日韩高清| 在线观看日韩www视频免费| 一区二区视频免费在线观看| 久久久精品网| 久久精品66| 韩国av网站在线| 韩国中文字幕av| 好看的日韩精品| 久久精品国产精品| 色久优优欧美色久优优| 成人综合在线视频| 国产精品毛片一区二区在线看| 欧美办公室脚交xxxx| 免费h片在线| 欧美大片免费播放| 国产精品视频午夜| 亚洲一二三在线| 色呦呦国产精品| 福利视频网站一区二区三区| 伊人久久大香线蕉精品组织观看| 日韩一区二区三区免费视频| 精品推荐蜜桃传媒| www.日本xxxx| 91香蕉视频网址| 97欧洲一区二区精品免费| 欧美成人国产va精品日本一级| 欧美一区二区播放|