Android進(jìn)階(四):Activity啟動(dòng)過(guò)程(最詳細(xì)&最簡(jiǎn)單)

1.前言

  • 最近一直在看 《Android進(jìn)階解密》 的一本書(shū)潜秋,這本書(shū)編寫(xiě)邏輯蛔琅、流程都非常好,而且很容易看懂半等,非常推薦大家去看看(沒(méi)有收廣告費(fèi)揍愁,單純覺(jué)得作者寫(xiě)的很好)。
  • 上一篇簡(jiǎn)單的介紹了Android進(jìn)階(三):Application啟動(dòng)過(guò)程(最詳細(xì)&最簡(jiǎn)單)杀饵。
  • 今天就介紹Application啟動(dòng)之后,是如何啟動(dòng)應(yīng)用程序中的 第一個(gè)Activity (基于Android 8.0 系統(tǒng))谬擦。
  • 文章中實(shí)例 linhaojian的Github

2.Activity啟動(dòng)過(guò)程的時(shí)序圖

Activity啟動(dòng)流程.png

3.源碼分析

3.1 ActivityManagerService

private final boolean attachApplicationLocked(IApplicationThread thread,
      int pid, int callingUid, long startSeq) {
    //...
    //通知ActivityThread啟動(dòng)application
    thread.bindApplication(processName, appInfo, providers, null, profilerInfo,
                    null, null, null, testMode,
                    mBinderTransactionTrackingEnabled, enableTrackAllocation,
                    isRestrictedBackupMode || !normalMode, app.persistent,
                    new Configuration(getGlobalConfiguration()), app.compat,
                    getCommonServicesLocked(app.isolated),
                    mCoreSettingsObserver.getCoreSettingsLocked(),
                    buildSerial, isAutofillCompatEnabled);// 1
     // 啟動(dòng)第一個(gè)Activity
    if (normalMode) {
        try {
            //Application初始化之后切距,初始化&啟動(dòng)第一個(gè)Activity
            if (mStackSupervisor.attachApplicationLocked(app)) {// 2
                didSomething = true;
            }
        } catch (Exception e) {
            Slog.wtf(TAG, "Exception thrown launching activities in " + app, e);
            badApp = true;
        }
    }     
}
  • 注釋1:通知ActivityThread啟動(dòng)application
  • 注釋2:調(diào)用ActivityStackSupervisor惨远,啟動(dòng)第一個(gè)Activity谜悟;

3.2 ActivityStackSupervisor 啟動(dòng)Activity

    boolean attachApplicationLocked(ProcessRecord app) throws RemoteException {
        //獲取應(yīng)用的進(jìn)行名稱
        final String processName = app.processName;
        boolean didSomething = false;
        for (int displayNdx = mActivityDisplays.size() - 1; displayNdx >= 0; --displayNdx) {
            final ActivityDisplay display = mActivityDisplays.valueAt(displayNdx);
            for (int stackNdx = display.getChildCount() - 1; stackNdx >= 0; --stackNdx) {
                final ActivityStack stack = display.getChildAt(stackNdx);
                if (!isFocusedStack(stack)) {
                    continue;
                }
                stack.getAllRunningVisibleActivitiesLocked(mTmpActivityList);
                final ActivityRecord top = stack.topRunningActivityLocked();
                final int size = mTmpActivityList.size();
                for (int i = 0; i < size; i++) {
                    final ActivityRecord activity = mTmpActivityList.get(i);
                    if (activity.app == null && app.uid == activity.info.applicationInfo.uid
                            && processName.equals(activity.processName)) {
                        try {
                            // Application初始化之后,啟動(dòng)第一個(gè)Activity
                            if (realStartActivityLocked(activity, app,
                                    top == activity /* andResume */, true /* checkConfig */)) { // 1
                                didSomething = true;
                            }
                        } catch (RemoteException e) {
                            Slog.w(TAG, "Exception in new application when starting activity "
                                    + top.intent.getComponent().flattenToShortString(), e);
                            throw e;
                        }
                    }
                }
            }
        }
        if (!didSomething) {
            ensureActivitiesVisibleLocked(null, 0, !PRESERVE_WINDOWS);
        }
        return didSomething;
    }
  • 注釋1:調(diào)用自身的realStartActivityLocked()北秽,真正的啟動(dòng)Activity葡幸;
    final boolean realStartActivityLocked(ActivityRecord r, ProcessRecord app,
            boolean andResume, boolean checkConfig) throws RemoteException {
                //...
                // Create activity launch transaction.
                //通知ActivityThread創(chuàng)建activity的實(shí)例 ;調(diào)用Activity的OnCreate贺氓; 創(chuàng)建對(duì)應(yīng)PhoneWindow實(shí)例
                final ClientTransaction clientTransaction = ClientTransaction.obtain(app.thread,
                        r.appToken);
                clientTransaction.addCallback(LaunchActivityItem.obtain(new Intent(r.intent),
                        System.identityHashCode(r), r.info,
                        // TODO: Have this take the merged configuration instead of separate global
                        // and override configs.
                        mergedConfiguration.getGlobalConfiguration(),
                        mergedConfiguration.getOverrideConfiguration(), r.compat,
                        r.launchedFromPackage, task.voiceInteractor, app.repProcState, r.icicle,
                        r.persistentState, results, newIntents, mService.isNextTransitionForward(),
                        profilerInfo)); // 2
                // Set desired final state.
                final ActivityLifecycleItem lifecycleItem;
                if (andResume) {
                    lifecycleItem = ResumeActivityItem.obtain(mService.isNextTransitionForward()); // 3
                } else {
                    lifecycleItem = PauseActivityItem.obtain(); // 4
                }
                clientTransaction.setLifecycleStateRequest(lifecycleItem);
                // 編制啟動(dòng)Activity實(shí)例的事務(wù)蔚叨,并按順序調(diào)用onCreate();onStart();onResume();
                mService.getLifecycleManager().scheduleTransaction(clientTransaction);// 5
    }
  • 注釋2:封裝啟動(dòng)Activity的相關(guān)數(shù)據(jù)與Activity的啟動(dòng)流程(LaunchActivityItem 下面會(huì)介紹該類)
  • 注釋3:封裝Activity啟動(dòng)之后生命周期變化為Resume的流程(ResumeActivityItem下面會(huì)介紹該類)辙培;
  • 注釋4:封裝Activity啟動(dòng)之后生命周期變化為Pause的流程蔑水;
  • 注釋5:真正開(kāi)始啟動(dòng)Activity與調(diào)用相關(guān)的生命周期方法

3.3 ClientLifecycleManager管理ClientTransaction

    void scheduleTransaction(ClientTransaction transaction) throws RemoteException {
        final IApplicationThread client = transaction.getClient();// 1
        transaction.schedule();// 2
        if (!(client instanceof Binder)) {
            // If client is not an instance of Binder - it's a remote call and at this point it is
            // safe to recycle the object. All objects used for local calls will be recycled after
            // the transaction is executed on client in ActivityThread.
            transaction.recycle();
        }
    }
  • 注釋1:獲取需要啟動(dòng)Activity進(jìn)程的代理對(duì)象IApplicationThread扬蕊;
  • 注釋2:調(diào)用ClientTransaction中的schedule()搀别;

3.4 ClientTransaction中schedule函數(shù)

    public void schedule() throws RemoteException {
        mClient.scheduleTransaction(this);// 1
    }
  • 注釋1:mClient就是與ActivityThread通訊的代理對(duì)象(IApplicationThread),所以這里其實(shí)是調(diào)用ActivityThread類中ApplicationThread內(nèi)部類的scheduleTransaction()尾抑;

3.5 ApplicationThread的scheduleTransaction函數(shù)

    private class ApplicationThread extends IApplicationThread.Stub {
        @Override
        public void scheduleTransaction(ClientTransaction transaction) throws RemoteException {
            ActivityThread.this.scheduleTransaction(transaction);// 1
        }    
    }
  • 注釋1:這里其實(shí)最后還是調(diào)用了ActivityThread的scheduleTransaction()歇父,而這個(gè)scheduleTransaction()其實(shí)真正的實(shí)現(xiàn)是ClientTransactionHandler蒂培,因?yàn)锳ctivityThread是繼承ClientTransactionHandler類的
public abstract class ClientTransactionHandler {
    /** Prepare and schedule transaction for execution. */
    void scheduleTransaction(ClientTransaction transaction) {
        transaction.preExecute(this);
        sendMessage(ActivityThread.H.EXECUTE_TRANSACTION, transaction);// 2
    }
}
  • 注釋2:調(diào)用ActivityThread的sendMessage()榜苫,將ClientTransaction參數(shù)通過(guò)Handler機(jī)制切換至主線程進(jìn)行處理毁渗;

3.6 ActivityThread的H類

    class H extends Handler {
        //...
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case EXECUTE_TRANSACTION: // 1
                    final ClientTransaction transaction = (ClientTransaction) msg.obj;// 2
                    mTransactionExecutor.execute(transaction);// 3
                    if (isSystem()) {
                        // Client transactions inside system process are recycled on the client side
                        // instead of ClientLifecycleManager to avoid being cleared before this
                        // message is handled.
                        transaction.recycle();
                    }
                    // TODO(lifecycler): Recycle locally scheduled transactions.
                    break;              
            }        
        }
    }
  • 注釋1:接收ClientTransactionHandler發(fā)送的EXECUTE_TRANSACTION事件
  • 注釋2:接收ClientTransactionHandler發(fā)送的ClientTransaction參數(shù)单刁;
  • 注釋3:通過(guò)TransactionExecutor真正處理ClientTransaction中封裝的Activity相關(guān)信息灸异;

3.7 TransactionExecutor處理ClientTransaction

    public void execute(ClientTransaction transaction) {
        final IBinder token = transaction.getActivityToken();
        log("Start resolving transaction for client: " + mTransactionHandler + ", token: " + token);
        executeCallbacks(transaction);
        executeLifecycleState(transaction);
        mPendingActions.clear();
        log("End resolving transaction");
    }

3.8 executeCallbacks()啟動(dòng)Activity

    @VisibleForTesting
    public void executeCallbacks(ClientTransaction transaction) {
        final List<ClientTransactionItem> callbacks = transaction.getCallbacks();
        if (callbacks == null) {
            // No callbacks to execute, return early.
            return;
        }
        log("Resolving callbacks");
        final IBinder token = transaction.getActivityToken();
        ActivityClientRecord r = mTransactionHandler.getActivityClient(token);
        // In case when post-execution state of the last callback matches the final state requested
        // for the activity in this transaction, we won't do the last transition here and do it when
        // moving to final state instead (because it may contain additional parameters from server).
        final ActivityLifecycleItem finalStateRequest = transaction.getLifecycleStateRequest();
        final int finalState = finalStateRequest != null ? finalStateRequest.getTargetState()
                : UNDEFINED;
        // Index of the last callback that requests some post-execution state.
        final int lastCallbackRequestingState = lastCallbackRequestingState(transaction);
        final int size = callbacks.size();
        for (int i = 0; i < size; ++i) {
            final ClientTransactionItem item = callbacks.get(i);// 1
            log("Resolving callback: " + item);
            final int postExecutionState = item.getPostExecutionState();
            final int closestPreExecutionState = mHelper.getClosestPreExecutionState(r,
                    item.getPostExecutionState());
            if (closestPreExecutionState != UNDEFINED) {
                cycleToPath(r, closestPreExecutionState);
            }
            item.execute(mTransactionHandler, token, mPendingActions);// 2
            item.postExecute(mTransactionHandler, token, mPendingActions);
            if (r == null) {
                // Launch activity request will create an activity record.
                r = mTransactionHandler.getActivityClient(token);
            }
            if (postExecutionState != UNDEFINED && r != null) {
                // Skip the very last transition and perform it by explicit state request instead.
                final boolean shouldExcludeLastTransition =
                        i == lastCallbackRequestingState && finalState == postExecutionState;
                cycleToPath(r, postExecutionState, shouldExcludeLastTransition);
            }
        }
    }
  • 注釋1:獲取ClientTransaction中的ClientTransactionItem對(duì)象(其實(shí)這里的對(duì)象就上面3.2介紹中的LaunchActivityItem)
  • 注釋2:調(diào)用LaunchActivityItem類的execute()羔飞,下面我們看看LaunchActivityItem肺樟;
public class LaunchActivityItem extends ClientTransactionItem {
    @Override
    public void execute(ClientTransactionHandler client, IBinder token,// 1
            PendingTransactionActions pendingActions) {
        Trace.traceBegin(TRACE_TAG_ACTIVITY_MANAGER, "activityStart");
        ActivityClientRecord r = new ActivityClientRecord(token, mIntent, mIdent, mInfo,
                mOverrideConfig, mCompatInfo, mReferrer, mVoiceInteractor, mState, mPersistentState,
                mPendingResults, mPendingNewIntents, mIsForward,
                mProfilerInfo, client);
        client.handleLaunchActivity(r, pendingActions, null /* customIntent */);// 2
        Trace.traceEnd(TRACE_TAG_ACTIVITY_MANAGER);
    }
}
  • 注釋1:剛剛說(shuō)了ClientTransactionHandler是被ActivityThread繼承,所以這里傳入進(jìn)來(lái)就是ActivityThread;
  • 注釋2:看到了吧逻淌,經(jīng)過(guò)多次不同層次相互調(diào)用么伯,最終真正調(diào)用ActivityThread的handleLaunchActivity(),啟動(dòng)應(yīng)用的第一個(gè)Activity;

3.8 executeLifecycleState()改變Activity的生命周期

    private void executeLifecycleState(ClientTransaction transaction) {
        final ActivityLifecycleItem lifecycleItem = transaction.getLifecycleStateRequest();// 1
        if (lifecycleItem == null) {
            // No lifecycle request, return early.
            return;
        }
        log("Resolving lifecycle state: " + lifecycleItem);
        final IBinder token = transaction.getActivityToken();
        final ActivityClientRecord r = mTransactionHandler.getActivityClient(token);
        if (r == null) {
            // Ignore requests for non-existent client records for now.
            return;
        }
        // Cycle to the state right before the final requested state.
        cycleToPath(r, lifecycleItem.getTargetState(), true /* excludeLastState */);
        // Execute the final transition with proper parameters.
        lifecycleItem.execute(mTransactionHandler, token, mPendingActions); // 2
        lifecycleItem.postExecute(mTransactionHandler, token, mPendingActions);
    }
  • 注釋1:獲取ClientTransaction中的ActivityLifecycleItem 對(duì)象(其實(shí)這里的對(duì)象就上面3.2介紹中的ResumeActivityItem)卡儒;
  • 注釋2:調(diào)用ResumeActivityItem的execute()田柔,下面我們看看ResumeActivityItem
public class ResumeActivityItem extends ActivityLifecycleItem {
    //...
    @Override
    public void execute(ClientTransactionHandler client, IBinder token,
            PendingTransactionActions pendingActions) {
        Trace.traceBegin(TRACE_TAG_ACTIVITY_MANAGER, "activityResume");
        client.handleResumeActivity(token, true /* finalStateRequest */, mIsForward, // 1
                "RESUME_ACTIVITY");
        Trace.traceEnd(TRACE_TAG_ACTIVITY_MANAGER);
    }
}
  • 注釋1:其實(shí)就是調(diào)用Activity的handleResumeActivity()骨望,最終就會(huì)調(diào)用Activity的onStart()與onResume()硬爆;

4.類關(guān)系

Activity啟動(dòng)類結(jié)構(gòu).png
  • 通過(guò)上圖發(fā)現(xiàn),Activity啟動(dòng)過(guò)程就是AMS與ActivityThread不同的進(jìn)程交互實(shí)現(xiàn)的擎鸠。

5.總結(jié)

  • 到此缀磕,Activity啟動(dòng)過(guò)程介紹完畢。
  • 如果喜歡我的分享劣光,可以點(diǎn)擊 關(guān)注 或者 袜蚕,你們支持是我分享的最大動(dòng)力 。
  • linhaojian的Github

歡迎關(guān)注linhaojian_CSDN博客或者linhaojian_簡(jiǎn)書(shū)绢涡!

不定期分享關(guān)于安卓開(kāi)發(fā)的干貨牲剃。


寫(xiě)技術(shù)文章初心

  • 技術(shù)知識(shí)積累
  • 技術(shù)知識(shí)鞏固
  • 技術(shù)知識(shí)分享
  • 技術(shù)知識(shí)交流
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市雄可,隨后出現(xiàn)的幾起案子凿傅,更是在濱河造成了極大的恐慌,老刑警劉巖滞项,帶你破解...
    沈念sama閱讀 221,576評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件狭归,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡文判,警方通過(guò)查閱死者的電腦和手機(jī)过椎,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,515評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)戏仓,“玉大人疚宇,你說(shuō)我怎么就攤上這事亡鼠。” “怎么了敷待?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,017評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵间涵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我榜揖,道長(zhǎng)勾哩,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,626評(píng)論 1 296
  • 正文 為了忘掉前任举哟,我火速辦了婚禮思劳,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘妨猩。我一直安慰自己潜叛,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,625評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布壶硅。 她就那樣靜靜地躺著威兜,像睡著了一般。 火紅的嫁衣襯著肌膚如雪庐椒。 梳的紋絲不亂的頭發(fā)上椒舵,一...
    開(kāi)封第一講書(shū)人閱讀 52,255評(píng)論 1 308
  • 那天,我揣著相機(jī)與錄音扼睬,去河邊找鬼逮栅。 笑死,一個(gè)胖子當(dāng)著我的面吹牛窗宇,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播特纤,決...
    沈念sama閱讀 40,825評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼军俊,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了捧存?” 一聲冷哼從身側(cè)響起粪躬,我...
    開(kāi)封第一講書(shū)人閱讀 39,729評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎昔穴,沒(méi)想到半個(gè)月后镰官,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,271評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡吗货,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,363評(píng)論 3 340
  • 正文 我和宋清朗相戀三年泳唠,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片宙搬。...
    茶點(diǎn)故事閱讀 40,498評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡笨腥,死狀恐怖拓哺,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情脖母,我是刑警寧澤士鸥,帶...
    沈念sama閱讀 36,183評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站谆级,受9級(jí)特大地震影響烤礁,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜肥照,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,867評(píng)論 3 333
  • 文/蒙蒙 一脚仔、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧建峭,春花似錦玻侥、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,338評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至边锁,卻和暖如春姑食,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背茅坛。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,458評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工音半, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人贡蓖。 一個(gè)月前我還...
    沈念sama閱讀 48,906評(píng)論 3 376
  • 正文 我出身青樓曹鸠,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親斥铺。 傳聞我的和親對(duì)象是個(gè)殘疾皇子彻桃,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,507評(píng)論 2 359

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