Android源碼分析——Activity的啟動過程

最近在看維術的Android插件化原理解析,需要補充一些Framework層的知識,首先來研究Activity的啟動過程。

Activity的啟動從Activity類中startActivity方法(先看Actiivty中的,后面再看Context中的,本質(zhì)是一樣的)開始。跟著這個方法一步一步跟蹤,會發(fā)現(xiàn)它最后在startActivityForResult里面調(diào)用了Instrument對象的execStartActivity方法;
Instrumentation類中的execStartActivity方法,源碼如下:

public ActivityResult execStartActivity(
            Context who, IBinder contextThread, IBinder token, Activity target,
            Intent intent, int requestCode, Bundle options) {
        IApplicationThread whoThread = (IApplicationThread) contextThread;
        ......
        ......
        try {
            intent.migrateExtraStreamToClipData();
            intent.prepareToLeaveProcess();
            int result = ActivityManagerNative.getDefault()
                .startActivity(whoThread, who.getBasePackageName(), intent,
                        intent.resolveTypeIfNeeded(who.getContentResolver()),
                        token, target != null ? target.mEmbeddedID : null,
                        requestCode, 0, null, options);
            checkStartActivityResult(result, intent);
        } catch (RemoteException e) {
        }
        return null;
    }

發(fā)現(xiàn)最終調(diào)用的是ActivityManagerNative類中的startActivity方法。ActivityManagerNative指的是Binder本地對象,這個類是抽象類,它的實現(xiàn)是ActivityManagerService;因此對于AMS的最終操作都會進入ActivityManagerService這個真正實現(xiàn),接著調(diào)用startActivityAsUser方法,最終調(diào)用ActivityStackSupervisor類中的startActivityMayWait方法,ActivityStackSupervisor這個類低版本沒有,在startActivityMayWait中調(diào)用startActivityLocked之后處理的都是Activity任務棧相關內(nèi)容,這一系列ActivityStack和ActivityStackSupervisor糾纏不清的調(diào)用看下圖就明白了;


調(diào)用流程圖

ActivityStack類中的resumeTopActivityLocked方法中調(diào)用ActivityStack中的resumeTopActivityInnerLocked,然后在改方法中調(diào)用ActivityStackSupervisor類中的startSpecificActivityLocked方法:

void startSpecificActivityLocked(ActivityRecord r,
            boolean andResume, boolean checkConfig) {
        // Is this activity's application already running?
        ProcessRecord app = mService.getProcessRecordLocked(r.processName,
                r.info.applicationInfo.uid, true);

        r.task.stack.setLaunchTime(r);
        //如果App已經(jīng)運行了,直接創(chuàng)建Activity
        if (app != null && app.thread != null) {
            try {
                if ((r.info.flags&ActivityInfo.FLAG_MULTIPROCESS) == 0
                        || !"android".equals(r.info.packageName)) {
                    // Don't add this if it is a platform component that is marked
                    // to run in multiple processes, because this is actually
                    // part of the framework so doesn't make sense to track as a
                    // separate apk in the process.
                    app.addPackage(r.info.packageName, r.info.applicationInfo.versionCode,
                            mService.mProcessStats);
                }
                //真正啟動Activity
                realStartActivityLocked(r, app, andResume, checkConfig);
                return;
            } catch (RemoteException e) {
                Slog.w(TAG, "Exception when starting activity "
                        + r.intent.getComponent().flattenToShortString(), e);
            }

            // If a dead object exception was thrown -- fall through to
            // restart the application.
        }
        //如果App還沒有運行,創(chuàng)建ActivityThread的進程,然后創(chuàng)建Activity
        mService.startProcessLocked(r.processName, r.info.applicationInfo, true, 0,
                "activity", r.intent.getComponent(), false, false, true);
        //走到這里會先創(chuàng)建ActivityThread的進程,在ActivityThread的main方法中調(diào)用attach方法,
        //接著調(diào)用AMS的attachApplication方法,接著調(diào)用AMS的attachApplicationLocked,
        //接著調(diào)用ActivityStackSupervisor的attachApplicationLocked,
        //接著調(diào)用realStartActivityLocked方法創(chuàng)建Activity
    }

接著在改方法中調(diào)用realStartActivityLocked,人如其名,這個方法開始了真正的“啟動Activity”:它調(diào)用了ApplicationThread的scheduleLaunchActivity方法,開始了真正的Activity對象創(chuàng)建以及啟動過程。

ApplicationThread是ActivityThread中定義的內(nèi)部類,繼承ApplicationThreadNative(ApplicationThreadNative extends Binder implements IApplicationThread ),ApplicationThread實際上是一個Binder對象,是App所在的進程與AMS所在進程system_server通信的橋梁。

這里的scheduleLaunchActivity方法直接把啟動Activity的任務通過一個消息轉(zhuǎn)發(fā)給了主線程;我們查看Handler類對于這個消息的處理:

case LAUNCH_ACTIVITY: {
                    Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "activityStart");
                    final ActivityClientRecord r = (ActivityClientRecord) msg.obj;

                    r.packageInfo = getPackageInfoNoCheck(
                            r.activityInfo.applicationInfo, r.compatInfo);
                    handleLaunchActivity(r, null);
                    Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
                } break;

可以看到,這里直接調(diào)用了ActivityThread的handleLaunchActivity方法,在這個方法內(nèi)部有一句非常重要:

Activity a = performLaunchActivity(r, customIntent);

繞了這么多彎,我們的Activity終于被創(chuàng)建出來了!這個方法做了兩件很重要的事情:
使用ClassLoader加載并通過反射創(chuàng)建Activity對象:

java.lang.ClassLoader cl = r.packageInfo.getClassLoader();
activity = mInstrumentation.newActivity(cl, component.getClassName(), r.intent);
StrictMode.incrementExpectedActivityCount(activity.getClass());
r.intent.setExtrasClassLoader(cl);
r.intent.prepareToEnterProcess();
if (r.state != null) {
    r.state.setClassLoader(cl);
}

如果Application還沒有創(chuàng)建,那么創(chuàng)建Application對象并回調(diào)相應的生命周期方法;

Application app = r.packageInfo.makeApplication(false, mInstrumentation);

Activity的啟動過程到這里就結(jié)束了,這里調(diào)用了一系列方法,大多數(shù)方法我也不知道是干嘛的,留待以后研究,畢竟代碼邏輯還是很復雜的。

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容