源码分析 Activity 可见性真实时机

本文通过探索 Android 生命周期方法的源码,寻找 Activity 中 DecorView 绘制的真正时机。

Blog-Activity可见性真实时机源码分析

背景

最初对生命周期中 Activity "可见"的理解是:相关 View 绘制完毕,并且可以从屏幕上看到。而网上很多文章中说 onStart(onResume)时可见。

最近阅读源码逻辑发现与上述结论有矛盾:Activity 的 DecorView 真正的绘制发生在 onResume 方法之后。

那么到底哪个是对的呢?

背景知识

WindowManager 通过 IPC 通信将 View 交给 WindowManagerService(WMS),WMS 继而创建 SurfaceSession 完成窗口的绘制

源码分析

通过分别查找生命周期方法与绘制方法,将俩者触发顺序进行比较,找到可见在 Activity 生命周期里的真正时机。

生命周期方法的触发时机

这里直接分析 Android 系统通过 AMS 启动 ActivityThread 之后,ActivityThread 继而启动 Activity 的过程。以下为 8.0 源码。

onCreate 触发时机

ActivityThread 首先会调用 handleLaunchActivity() 方法:

private void handleLaunchActivity(ActivityClientRecord r, Intent customIntent, String reason) {
        ...

        Activity a = performLaunchActivity(r, customIntent); //1.

        if (a != null) {
            ...
            handleResumeActivity(r.token, false, r.isForward,
                    !r.activity.mFinished && !r.startsNotResumed, r.lastProcessedSeq, reason); //2.

handleLaunchActivity() 执行了注释1、2处的俩个关键方法。这里先分析注释1处的 performLaunchActivity:

private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {
...
mInstrumentation.callActivityOnCreate(activity, r.state, r.persistentState);
...

performLaunchActivity 是 Activity 启动过程中非常重要的方法,它创建了 Context、Activtiy 等关键类,但这不是分析的重点。重点是在初始化这些关键类之后,会调用mInstrumentation.callActivityOnCreate()执行 Activity 的 onCreate 方法:

    public void callActivityOnCreate(Activity activity, Bundle icicle,
            PersistableBundle persistentState) {
        prePerformCreate(activity);
        activity.performCreate(icicle, persistentState);
        postPerformCreate(activity);
    }

看下 activity.performCreate

    final void performCreate(Bundle icicle, PersistableBundle persistentState) {
        mCanEnterPictureInPicture = true;
        restoreHasCurrentPermissionRequest(icicle);
        if (persistentState != null) {
            onCreate(icicle, persistentState);
        } else {
            onCreate(icicle);
        }
        ...
    }

所以 onCreate 的触发时机为:

ActivityThread.handleLaunchActivity -> ActivityThread.performLaunchActivity -> Instrumentation.callActivityOnCreate -> Activity.performCreate

onCreate.png set-w320

onRestart & onStart 触发时机

接着看 handleLaunchActivity 源码中的注释2 -- handleResumeActivity :

    final void handleResumeActivity(IBinder token,
            boolean clearHide, boolean isForward, boolean reallyResume, int seq, String reason) {
        ActivityClientRecord r = mActivities.get(token);
        ...

        //1.
        r = performResumeActivity(token, clearHide, reason);

        if (r != null) {
            final Activity a = r.activity;
            ...
            //2.
            if (r.window == null && !a.mFinished && willBeVisible) {
                r.window = r.activity.getWindow();
                View decor = r.window.getDecorView();
                decor.setVisibility(View.INVISIBLE);
                ViewManager wm = a.getWindowManager();
                WindowManager.LayoutParams l = r.window.getAttributes();
                a.mDecor = decor;
                l.type = WindowManager.LayoutParams.TYPE_BASE_APPLICATION;
                l.softInputMode |= forwardBit;
                if (r.mPreserveWindow) {
                    a.mWindowAdded = true;
                    r.mPreserveWindow = false;
                    ViewRootImpl impl = decor.getViewRootImpl();
                    if (impl != null) {
                        impl.notifyChildRebuilt();
                    }
                }
                if (a.mVisibleFromClient) {
                    if (!a.mWindowAdded) {
                        a.mWindowAdded = true;
                        //3.
                        wm.addView(decor, l);
                    } else {
                        a.onWindowAttributesChanged(l);
                    }
                }
                ...

这里只关注注释1处,首先调用了 performResumeActivity() 方法,而 performResumeActivity() 紧接着又调用了 Activity.performResume() :

final void performResume(boolean followedByPause, String reason) {
    performRestart(true /* start */, reason); //1.
    ...
    mInstrumentation.callActivityOnResume(this); //2.
    ...
    onPostResume();
    ...
}

先看 performRestart() 源码 :

    final void performRestart(boolean start, String reason) {
        ...

        if (mToken != null && mParent == null) {
            WindowManagerGlobal.getInstance().setStoppedState(mToken, false /* stopped */);
        }

        if (mStopped) {
            mStopped = false;

            synchronized (mManagedCursors) {
                final int N = mManagedCursors.size();
                for (int i=0; i<N; i++) {
                    ManagedCursor mc = mManagedCursors.get(i);
                    if (mc.mReleased || mc.mUpdated) {
                        if (!mc.mCursor.requery()) {
                            if (getApplicationInfo().targetSdkVersion
                                    >= android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
                                throw new IllegalStateException(
                                        "trying to requery an already closed cursor  "
                                        + mc.mCursor);
                            }
                        }
                        mc.mReleased = false;
                        mc.mUpdated = false;
                    }
                }
            }

            mCalled = false;
            mInstrumentation.callActivityOnRestart(this); //1.
            writeEventLog(LOG_AM_ON_RESTART_CALLED, reason);
            ...
            if (start) {
                performStart(reason); // 2.
            }
        }
    }

注释1处,最终执行了 Activity.onRestart() 方法。
注释2处,调用 performStart :

 final void performStart(String reason) {
        mActivityTransitionState.setEnterActivityOptions(this, getActivityOptions());
        mFragments.noteStateNotSaved();
        mCalled = false;
        mFragments.execPendingActions();
        mInstrumentation.callActivityOnStart(this); //1.
        ...

performStart() 调用了 mInstrumentation.callActivityOnStart(this),它最终调用了 Activity.onStart()。

所以 onRestart 的触发时机为:

ActivityThread.handleLaunchActivity -> ActivityThread.handleResumeActivity -> ActivityThread.performResumeActivity -> Activity.performResume -> Activity.performRestart -> Instrumentation.callActivityOnRestart

onRestart.png

onStart 的触发时机为:

ActivityThread.handleLaunchActivity -> ActivityThread.handleResumeActivity -> ActivityThread.performResumeActivity -> Activity.performResume -> Activity.performRestart -> Activity.performStart -> Instrumentation.callActivityOnStart

onStart.png

onResume 触发时机

回到上面的 performResume() 方法,performRestart() 执行完 onRestart()、onStart() 之后,紧接着调用了 Instrumentation.callActivityOnResume(this) 去执行 onResume() 方法。

onResume 的触发时机为:

ActivityThread.handleLaunchActivity -> ActivityThread.handleResumeActivity -> ActivityThread.performResumeActivity -> Activity.performResume -> Instrumentation.callActivityOnResume

onResume.png

绘制的触发时机

真实绘制发生在 WMS 所在的系统进程中,所以只要找到对应应用层的 WM 是什么时候添加的 DecorView 即可。

由上面的分析我们可知:

performResumeActivity() 方法分别执行了 onRestart()、onStart()、onResume
() 等生命周期函数。而在 handleResumeActivity() 源码中,我们可以看到在 performResumeActivity() 执行完毕后,才做了真实绘制操作:

    final void handleResumeActivity(IBinder token,
            boolean clearHide, boolean isForward, boolean reallyResume, int seq, String reason) {
        ActivityClientRecord r = mActivities.get(token);
        ...

        //1.
        r = performResumeActivity(token, clearHide, reason);

        if (r != null) {
            final Activity a = r.activity;
            ...
            //2.
            if (r.window == null && !a.mFinished && willBeVisible) {
                r.window = r.activity.getWindow();
                View decor = r.window.getDecorView();
                decor.setVisibility(View.INVISIBLE);
                ViewManager wm = a.getWindowManager();
                WindowManager.LayoutParams l = r.window.getAttributes();
                a.mDecor = decor;
                l.type = WindowManager.LayoutParams.TYPE_BASE_APPLICATION;
                l.softInputMode |= forwardBit;
                if (r.mPreserveWindow) {
                    a.mWindowAdded = true;
                    r.mPreserveWindow = false;
                    ViewRootImpl impl = decor.getViewRootImpl();
                    if (impl != null) {
                        impl.notifyChildRebuilt();
                    }
                }
                if (a.mVisibleFromClient) {
                    if (!a.mWindowAdded) {
                        a.mWindowAdded = true;
                        //3.
                        wm.addView(decor, l);
                    } else {
                        a.onWindowAttributesChanged(l);
                    }
                }
                ...

上述注释2、3,将 DecorView 交给了 WMS 去绘制,而通过前面分析可知,这个操作发生在 onResume 方法之后。

执行顺序图

顺序图.png

总结

首先,handleResumeActivity() 不同于字面意思,它共计执行了 onRestart()、onStart()、onResume() 三个生命周期函数。

其次,wm.addView(decor, l) 发生在 handleResumeActivity() 之后,说明 View 的真正绘制发生在 onResume() 之后。而 onCreate() 中调用 setContentView() 只是创建 DecorView 并添加子视图,并没有真正开始绘制。

最后,有一些关键知识点需要了解,后续会专程研究,这里可以如下简单理解:

一个 View 从数据渲染为屏幕可见的视图,不是光靠应用层就能完成的。应用层只负责数据的处理,真正的渲染发生在系统层,俩者通过 Android 匿名共享内存进行通信。

应用层通过 ViewRootImpl(Surface),与系统层的 SurfaceFlinger 进行交互。其中 ViewRootImpl 可以简单理解为 View 在应用层数据的处理者,而 SurfaceFlinger 可以理解为 View 在系统层渲染的处理者。

WindowManager.addView 之后,相应窗口的 ViewRootImpl、以及对应系统层的 SurfaceSession 才会创建。后续添加、更新 View,都是通过 ViewRootImpl 触发 View 的三大方法,计算 View 数据,然后交给系统层 SurfaceFlinger 去绘制的。

所以上述 wm.addView(decor, l) 发生在 handleResumeActivity() 之后,可以得出 View 绘制发生在 onResume() 之后,就是基于此。

那么为什么网上很多文章都说 onStart(onResume)可见呢?上面说,应用层负责数据、系统层负责渲染。所以我的理解是:应用层对于 View 的可见性理解,是基于数据的,只要数据设置完毕,后续系统层级的绘制工作无需关心。而我这里的可见,是视觉意义上真正的绘制完毕,故不能抛开系统层的渲染,所以 Activity 中 DecorView 的可见发生在 onResume() 方法之后。

以上为个人理解,有异议欢迎留言。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,080评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,422评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,630评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,554评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,662评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,856评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,014评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,752评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,212评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,541评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,687评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,347评论 4 331
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,973评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,777评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,006评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,406评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,576评论 2 349

推荐阅读更多精彩内容