Android Activity生命周期,启动模式,启动过程详解

待兔
• 阅读 1614

前言

接触过Android开发的同学都知道Activity,Activity作为Android四大组件之一,使用频率高。简单来说Activity提供了一个显示界面,让用户进行各种操作,本文主要分为以下三个部分:Activity的生命周期,启动模式,以及Activity的工作过程。文中大部分篇幅来自《Android开发艺术探索》一书,尽管想多以流程或图片的形式说明,奈何文笔有限,Android系统整体把握还不够深入,只能借鉴参考前辈的成果,本文主要目的是帮自己梳理Activity的相关问题,疏漏之处欢迎指出。

说明:本文中的源码都是基于Android-25版本。

Activity生命周期

生命周期

首先给出Activity的完整生命周期,如下图所示:

Android Activity生命周期,启动模式,启动过程详解

Activity生命周期

图片来源Android开发者官网

看完了完整的生命周期,我们再分别简单介绍单个生命周期含义。

  • onCreate:表示Activity正在被创建,做一些初始化工作,如加载布局文件,对一些变量或者控件进行初始化
  • onStart:表示Activity正在启动,此时Activity已经可见了,但还无法和用户交互
  • onResume:表示Activity已经在前台,可以和用户交互了
  • onPause:表示Activity正在停止,此时可以进行一些轻量级的存储数据和停止动画等工作,但不能太耗时,否则会影响到新Activity的显示,Android指定如果onPause在500ms内没有执行完毕的话就会强制关闭Activity。某些极端情况下,这个时候Activity会回到当前Activity,此时onResume会被执行,很难重现这一情形
  • onStop:表示Activity即将停止,此时Activity已经不可见了,但是Activity对象还存在内存中,没有被销毁,这个阶段也主要做一些资源回收的工作,不能太耗时
  • onDestory:表示Activity即将被销毁,可以做一些回收工作和最终的资源释放,最后一个回调方法
  • onRestart:表示Activity重新启动,当前Activity从不可见重新变为可见时会调用onRestart,这种情况一般是用户的行为导致的,比如:用户按Home键切换到桌面或者打开了一个新的Activity

三个问题

onCreate,onStart和onResume之间的区别?

  • onCreate时Activity还不可见,onStart已经可见,还不在前台,onResume已经出现在前台
  • onCreate方法只在Activity创建时执行一次,而onStart方法在Activity的切换以及按Home键返回桌面再切回应用的过程中被多次调用,因此数据恢复在onStart方法中比较合适,onResume方法可以做一些开启动画和独占设备的操作

onPause,onStop和onDestroy之间的区别?

  • onPause时Activity还可见,onStop已经不可见,但Activity对象还存在内存中,onDestroy后Activity对象就不存在了
  • 内存不足时,可能不会执行onStop方法,因此程序状态保存,停止动画,存储数据等操作最好放在onPause

切换Activity时生命周期执行顺序

假设当前Activity为A,此时用户打开Activity B,那么A和B的生命周期的执行顺序为:

Android Activity生命周期,启动模式,启动过程详解

Activity跳转生命周期.png

为了验证生命周期执行顺序,我们写个例子实测,MainActivity中单击按钮跳转到SecondActivity,然后在相应的生命周期中输出日志,如下:

public class MainActivity extends Activity {

    private static final String TAG = "MainActivity";

    ... 

    @Override
    protected void onPause(){
        super.onPause();
        Log.d(TAG,"onPause");
    }

    @Override
    protected void onStop(){
        super.onStop();
        Log.d(TAG,"onStop");
    }
}

public class SecondActivity extends Activity {

    private static final String TAG = "SecondActivity";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Log.d(TAG,"onCreate");
    }

    @Override
    protected void onStart(){
        super.onStart();
        Log.d(TAG,"onStart");
    }

    @Override
    protected void onResume(){
        super.onResume();
        Log.d(TAG,"onResume");
    }
} 

日志结果如下:

08-31 12:36:36.086 3825-3825/com.sht.user.activityjumptest D/MainActivity: onPause
08-31 12:36:36.093 3825-3825/com.sht.user.activityjumptest D/SecondActivity: onCreate
08-31 12:36:36.093 3825-3825/com.sht.user.activityjumptest D/SecondActivity: onStart
08-31 12:36:36.094 3825-3825/com.sht.user.activityjumptest D/SecondActivity: onResume
08-31 12:36:36.517 3825-3825/com.sht.user.activityjumptest D/MainActivity: onStop 

那么为什么会有这样的生命周期执行顺序呢?Android官方文档对onPause的解释为:不能在onPause中做重量级的操作,因为必须onPause执行完成以后新Activity才能Resume,同时我在分析Activity启动源过程中码也得到了相同结论。

Activity保存状态

当 Activity 暂停或停止时,Activity 的状态会得到保留。因为当 Activity 暂停或停止时,Activity 对象仍保留在内存中 ,有关其成员和当前状态的所有信息仍处于活动状态。
但是当系统的配置发生变化(屏幕旋转,语言变换等)时,Activity 被销毁,系统回调 onSaveInstanceState() 方法对有关 Activity 状态的信息进行保存,系统会向该方法传递一个 Bundle,可以在其中使用 putString()putInt() 等方法以名称-值对形式保存有关 Activity 状态的信息。当 Activity 重建时将 Bundle 同时传递给 onCreate()onRestoreInstanceState() 方法,使用上述任一方法从 Bundle 提取保存的数据并恢复该 Activity 状态。

Android Activity生命周期,启动模式,启动过程详解

Activity保存状态

图片来源Android开发者官网

onSaveInstanceState()onRestoreInstanceState() 方法中,系统自动为我们做了一定的恢复工作,布局中的每个 View 调用自己的 onSaveInstanceState() 方法,让每个视图都能保存自身需保存的信息。
每个 View 都有 onSaveInstanceState()onRestoreInstanceState() 方法,具体能恢复哪些数据,需要查看其源码。保存和恢复 View 用到了事件委托思想,从下至上,再从上到下。需要注意的是,只有我们为 View 指定了唯一的ID属性(android:id)系统才会自动为其在需要时自动保存和恢复数据。
通过重写onSaveInstanceState()onRestoreInstanceState() 实现自定义的数据恢复机制。
那么 onSaveInstanceState()onRestoreInstanceState() 方法和 Activity 的生命周期的执行顺序是怎么样的?
onSaveInstanceState() 方法是在onStop之前,和onPause没有既定的时序关系, onRestoreInstanceState() 方法是在onStart之后、onResume之前调用的。

Activity启动模式

默认情况下,当我们多次启动同一个Activity时,系统会创建多个实例并把它们一一放入任务栈中,任务栈是一种后进先出的栈结构,每按下back键就会有一个Activity出栈,直到栈空为止。

standard:标准模式

这是系统默认的模式,每次启动一个新的Activity都会重新创建一个新的实例,不管这个实例已经是否存在。这种模式下,如果A启动了B(标准模式),那么B自动进入A所在的任务栈中。

singleTop:栈顶复用模式

此种模式下,如果新Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,同时onNewIntent方法会被调用,通过此方法的参数获取当前请求信息。而且,此Activity的onCreate,onStart不会被调用,因为没有发生改变。

singleTask:栈内复用模式

此种模式下,只要Activity在一个栈中存在,那么多次启动此Activity都不会重新创建实例,系统会回调onNewIntent方法。系统会先寻找是否存在A想要的任务栈,如果不存在,就重建一个任务栈,然后创建A的实例放入新任务栈中;如果存在A想要的任务栈,再查看是否有Activity实例存在,有的话就把该实例调到栈顶,如果实例不存在,则创建A的实例放入任务栈中。下面举三种例子说明此种模式运行机制:

  • 目前任务栈S1中为ABC,Activity D以singleTask模式请求启动,其需要的任务栈为S1,那么系统会创建D的实例,将D放入S1中
  • 目前任务栈S1中为ABC,Activity D以singleTask模式请求启动,其需要的任务栈为S2,那么系统会创建任务栈S2,再将D放入S2中
  • 目前任务栈S1中为ADBC,Activity D以singleTask模式请求启动,其需要的任务栈为S1,那么系统不会创建D的实例,将D切换到栈顶并调用其onNewIntent方法,同时栈内所有在D上面的Activity都需要出栈,最终的S1为AD

singleInstance:单实例模式

这是一种加强的singleTask模式,除了具有singleTask模式的所有特性外,具有此种模式的Activity只能单独位于一个任务栈中。比如Activity A以singleInstance模式启动,系统会为其创建一个新的任务栈,然后A独自运行在该任务栈中,后续的请求均不会创建新的Activity。

任务栈

在启动模式中多次提到Activity的任务栈,默认情况下,所有Activity所需任务栈的名字为应用的包名。此外还有一个参数TaskAffinity,这个参数标识了Activity所需要的任务栈的名字,TaskAffinity属性主要和singleTask模式或者allowTaskReparenting属性配对使用,具体请参考《Android开发艺术探索》P19Android任务和返回栈完全解析,细数那些你所不知道的细节

如何指定Activity启动模式

  • 第一种是通过AndroidManifest.xml 指定:
<activity android:name=".SecondActivity"
    android:launchMode="singleTask"
    android:label="@string/app_name"/> 
  • 第二种是通过在Intent中设置标志位,比如:
Intent intent = new Intent();
intent.setClass(MainActivity.this, SecondActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent); 

Activity启动过程源码分析

聊完了Activity的生命周期及启动模式,最头疼的部分来了,那就是一个Activity究竟是如何启动的呢?以下篇幅结合源码一探究竟,拨开云雾见青天。在显式调用启动Activity时,我们通过以下代码完成:

Intent intent = new Intent(this, TestActivity.class);
startActivity(intent); 

上述代码便可以启动一个Activity,下面我们便从源头理清Activity的启动流程,当然我们不可能对所有细节描述清楚,需要做的是理清流程,点到为止。

startActivity/startActivityForResult

我们从Activity的startActivity方法开始分析,读源码可知,startActivity方法有好几种重载形式,但最终都调用了startActivityForResult方法,因此我们重点分析startActivityForResult方法:

public void startActivityForResult(@RequiresPermission Intent intent, int requestCode, @Nullable Bundle options) {
    if (mParent == null) {
        options = transferSpringboardActivityOptions(options);
        Instrumentation.ActivityResult ar =
            mInstrumentation.execStartActivity(
                this, mMainThread.getApplicationThread(), mToken, this,
                intent, requestCode, options);
        if (ar != null) {
            mMainThread.sendActivityResult(
                mToken, mEmbeddedID, requestCode, ar.getResultCode(),
                ar.getResultData());
        }
        if (requestCode >= 0) {
            // If this start is requesting a result, we can avoid making
            // the activity visible until the result is received.  Setting
            // this code during onCreate(Bundle savedInstanceState) or onResume() will keep the
            // activity hidden during this time, to avoid flickering.
            // This can only be done when a result is requested because
            // that guarantees we will get information back when the
            // activity is finished, no matter what happens to it.
            mStartedActivity = true;
        }
        cancelInputsAndStartExitTransition(options);
        // TODO Consider clearing/flushing other event sources and events for child windows.
    } else {
        if (options != null) {
            mParent.startActivityFromChild(this, intent, requestCode, options);
        } else {
            // Note we want to go through this method for compatibility with
            // existing applications that may have overridden it.
            mParent.startActivityFromChild(this, intent, requestCode);
        }
    }
} 

Instrumentation

我们主要关注mParent == null这一部分逻辑,首先,mParent是一个Activity对象,表示该Activity是否由父Activity启动,如果该Activity是第一个被启动的,那么就会调用InstrumentationexecStartActivity方法:

public ActivityResult execStartActivity(
        Context who, IBinder contextThread, IBinder token, Activity target,
        Intent intent, int requestCode, Bundle options) {
    IApplicationThread whoThread = (IApplicationThread) contextThread;
    Uri referrer = target != null ? target.onProvideReferrer() : null;
    if (referrer != null) {
        intent.putExtra(Intent.EXTRA_REFERRER, referrer);
    }
    if (mActivityMonitors != null) {
        synchronized (mSync) {
            final int N = mActivityMonitors.size();
            for (int i=0; i<N; i++) {
                final ActivityMonitor am = mActivityMonitors.get(i);
                if (am.match(who, null, intent)) {
                    am.mHits++;
                    if (am.isBlocking()) {
                        return requestCode >= 0 ? am.getResult() : null;
                    }
                    break;
                }
            }
        }
    }
    try {
        intent.migrateExtraStreamToClipData();
        intent.prepareToLeaveProcess(who);
        int result = ActivityManagerNative.getDefault()
            .startActivity(whoThread, who.getBasePackageName(), intent,
                    intent.resolveTypeIfNeeded(who.getContentResolver()),
                    token, target != null ? target.mEmbeddedID : null,
                    requestCode, 0, null, options);
        // 检查启动Activity的结果
        checkStartActivityResult(result, intent);
    } catch (RemoteException e) {
        throw new RuntimeException("Failure from system", e);
    }
    return null;
} 

Instrumentation是管理Activity的一个工具类,包括创建和启动Activity,Activity的生命周期方法都是由Instrumentation这个类来控制,一个进程中只用一个Instrumentation实例。
execStartActivity是真正启动Activity的操作,下面解释其参数:

  • who : 待启动的Activity对象
  • mMainThread.getApplicationThread() : ApplicationThread对象,也是Binder对象
  • mToken :Binder对象,指向服务端一个ActivityRecord对象
  • target :待启动的Activity对象
  • intent :启动的intent对象
  • requestCode :请求码
  • options :参数

AMS

再回到execStartActivity方法体中,其中可以看到真正实现Activity启动的是ActivityManagerNative.getDefault().startActivity方法。下面列出了ActivityManagerNative.getDefault()的源码:

static public IActivityManager getDefault() {
    return gDefault.get();
}

private static final Singleton<IActivityManager> gDefault = new Singleton<IActivityManager>() {
    protected IActivityManager create() {
        IBinder b = ServiceManager.getService("activity");
        if (false) {
            Log.v("ActivityManager", "default service binder = " + b);
        }
        IActivityManager am = asInterface(b);
        if (false) {
            Log.v("ActivityManager", "default service = " + am);
        }
        return am;
    }
};

static public IActivityManager asInterface(IBinder obj) {
    if (obj == null) {
        return null;
    }
    IActivityManager in = 
        (IActivityManager)obj.queryLocalInterface(descriptor);
    if (in != null) {
        return in;                        // 同一进程,返回本地对象stub
    }
    return new ActivityManagerProxy(obj); // 跨进程,返回代理对象
} 

其中声明了一个Singleton封装类,类型是IActivityManager,第一次调用它的get方法它会通过create方法来初始化AMS这个Binder对象。其中在asInterface方法中返回了一个IActivityManager类型对象。
ActivityManagerService(AMS)继承自ActivityManagerNative,而ActivityManagerNative继承自Binder并实现了IActivityManager接口,因此AMS是一个Binder,同时也是IActivityManager的具体实现,ActivityManagerNative.getDefault()其实是一个IActivityManager类型的Binder对象,即AMS。我们理一下上面提到的各种类和接口之间的关系:

public final class ActivityManagerService extends ActivityManagerNative {...}

public abstract class ActivityManagerNative extends Binder implements IActivityManager {...}

public interface IActivityManager extends IInterface {...} 

所以Activity的启动过程转移到AMS中,下面分析AMSstartActivity方法:

public final int startActivity(IApplicationThread caller, String callingPackage,
        Intent intent, String resolvedType, IBinder resultTo, String resultWho, int requestCode,
        int startFlags, ProfilerInfo profilerInfo, Bundle bOptions) {
    return startActivityAsUser(caller, callingPackage, intent, resolvedType, resultTo,
            resultWho, requestCode, startFlags, profilerInfo, bOptions,
            UserHandle.getCallingUserId());
}

public final int startActivityAsUser(IApplicationThread caller, String callingPackage,
        Intent intent, String resolvedType, IBinder resultTo, String resultWho, int requestCode,
        int startFlags, ProfilerInfo profilerInfo, Bundle bOptions, int userId) {
    enforceNotIsolatedCaller("startActivity");
    userId = mUserController.handleIncomingUser(Binder.getCallingPid(), Binder.getCallingUid(),
            userId, false, ALLOW_FULL_ONLY, "startActivity", null);

    // Android7.0 Acitivty启动管理类新增ActivityStarter(原本是ActivityStackSupervisor处理该过程)
    return mActivityStarter.startActivityMayWait(caller, -1, callingPackage, intent,
            resolvedType, null, null, resultTo, resultWho, requestCode, startFlags,
            profilerInfo, null, null, bOptions, false, userId, null, null);
} 

ActivityStarter

上述代码中mActivityStarter是一个ActivityStarter对象,初始化为:

mActivityStarter = new ActivityStarter(this, mStackSupervisor); 

可以看出Activity的启动过程转移到ActivityStarterstartActivityMayWait方法中,startActivityMayWait又调用了startActivityLocked方法:

final int startActivityMayWait(IApplicationThread caller, int callingUid,
        String callingPackage, Intent intent, String resolvedType,
        IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
        IBinder resultTo, String resultWho, int requestCode, int startFlags,
        ProfilerInfo profilerInfo, IActivityManager.WaitResult outResult, Configuration config,
        Bundle bOptions, boolean ignoreTargetSecurity, int userId,
        IActivityContainer iContainer, TaskRecord inTask) {

        ...

        // 根据Intent中在系统中找到合适的应用的Activity,如果有多个Activity可选择则弹出界面让用户选择
        ActivityInfo aInfo = mSupervisor.resolveActivity(intent, rInfo, startFlags, profilerInfo);

        ...

        int res = startActivityLocked(caller, intent, ephemeralIntent, resolvedType,
                aInfo, rInfo, voiceSession, voiceInteractor,
                resultTo, resultWho, requestCode, callingPid,
                callingUid, callingPackage, realCallingPid, realCallingUid, startFlags,
                options, ignoreTargetSecurity, componentSpecified, outRecord, container,
                inTask);

        ...

        return res;
    }
} 

startActivityLocked方法又调用startActivityUnchecked方法:

final int startActivityLocked(IApplicationThread caller, Intent intent, Intent ephemeralIntent,
        String resolvedType, ActivityInfo aInfo, ResolveInfo rInfo,
        IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
        IBinder resultTo, String resultWho, int requestCode, int callingPid, int callingUid,
        String callingPackage, int realCallingPid, int realCallingUid, int startFlags,
        ActivityOptions options, boolean ignoreTargetSecurity, boolean componentSpecified,
        ActivityRecord[] outActivity, ActivityStackSupervisor.ActivityContainer container,
        TaskRecord inTask) {
    int err = ActivityManager.START_SUCCESS;

    ...

    // 创建ActivityRecord对象
    // ActivityRecord : 在AMS中,用ActivityRecord来作为Activity的记录者,每次启动一个Actvity会有一个对应
    // 的ActivityRecord对象,表示Activity的一个记录
    ActivityRecord r = new ActivityRecord(mService, callerApp, callingUid, callingPackage,
            intent, resolvedType, aInfo, mService.mConfiguration, resultRecord, resultWho,
            requestCode, componentSpecified, voiceSession != null, mSupervisor, container,
            options, sourceRecord);

    ...

    err = startActivityUnchecked(r, sourceRecord, voiceSession, voiceInteractor, startFlags,
            true, options, inTask);

    ...

    // 此处Activity对应的Task被移至前台
    postStartActivityUncheckedProcessing(r, err, stack.mStackId, mSourceRecord, mTargetStack);
    return err;
} 

startActivityUnchecked调用了ActivityStackstartActivityLocked方法和ActivityStackSupervisorresumeFocusedStackTopActivityLocked方法:

private int startActivityUnchecked(final ActivityRecord r, ActivityRecord sourceRecord,
        IVoiceInteractionSession voiceSession, IVoiceInteractor voiceInteractor,
        int startFlags, boolean doResume, ActivityOptions options, TaskRecord inTask) {

    ...

    // ActivityStack的startActivityLocked方法        
    mTargetStack.startActivityLocked(mStartActivity, newTask, mKeepCurTransition, mOptions);

    if (mDoResume) {
        final ActivityRecord topTaskActivity = mStartActivity.task.topRunningActivityLocked();
        if (!mTargetStack.isFocusable()
                || (topTaskActivity != null && topTaskActivity.mTaskOverlay
                && mStartActivity != topTaskActivity)) {
            // If the activity is not focusable, we can't resume it, but still would like to
            // make sure it becomes visible as it starts (this will also trigger entry
            // animation). An example of this are PIP activities.
            // Also, we don't want to resume activities in a task that currently has an overlay
            // as the starting activity just needs to be in the visible paused state until the
            // over is removed.
        } else {
            // ActivityStackSupervisor的resumeFocusedStackTopActivityLocked方法
            mSupervisor.resumeFocusedStackTopActivityLocked(mTargetStack, mStartActivity,mOptions);
        }

        ...
    }

    ...

} 

ActivityStackSupervisor

ActivityStackSupervisorresumeFocusedStackTopActivityLocked方法如下:

boolean resumeFocusedStackTopActivityLocked(
        ActivityStack targetStack, ActivityRecord target, ActivityOptions targetOptions) {
    if (targetStack != null && isFocusedStack(targetStack)) {
        // 待启动Activity对应的Task为前台栈时,调用该Task对应ActivityStack的resumeTopActivityUncheckedLocked方法
        return targetStack.resumeTopActivityUncheckedLocked(target, targetOptions);
    }
    final ActivityRecord r = mFocusedStack.topRunningActivityLocked();
    if (r == null || r.state != RESUMED) {
        // 否则只是调用此时前台栈的resumeTopActivityUncheckedLocked方法
        mFocusedStack.resumeTopActivityUncheckedLocked(null, null);
    }
    return false;
} 

ActivityStack

下面进入ActivityStackresumeTopActivityUncheckedLocked方法:

boolean resumeTopActivityUncheckedLocked(ActivityRecord prev, ActivityOptions options) {

    ...

    result = resumeTopActivityInnerLocked(prev, options);

    ...

    return result;
} 

紧接着又调用了resumeTopActivityInnerLocked方法:

private boolean resumeTopActivityInnerLocked(ActivityRecord prev, ActivityOptions options) {

    ...

    // mResumedActivity指向上一次启动的Activity
    if (mResumedActivity != null) {

        ...

        // 让上一次的Activity进入pause状态
        pausing |= startPausingLocked(userLeaving, false, next, dontWaitForPause);
    }

    if (pausing) {
        ...
        if (next.app != null && next.app.thread != null) {
            // 如果Activity所在App已经启动,则更新Activity所在进程优先级,防止被kill
            mService.updateLruProcessLocked(next.app, true, null);
        }
        ...
    } 

    ...

    if (next.app != null && next.app.thread != null) {
        ...
        if (next.newIntents != null) {
            // 如果Intent不为空,则调用NewIntent传入Intent参数
            next.app.thread.scheduleNewIntent(
                next.newIntents, next.appToken, false /* andPause */);
            ...
        }

        ...

        // Activity已经启动,重新进入前台
        next.app.thread.scheduleResumeActivity(next.appToken, next.app.repProcState,
            mService.isNextTransitionForward(), resumeAnimOptions);
        ...

    } else {
        ...
        // 创建进程,启动Activity,或者已经启动App,启动Activity
        mStackSupervisor.startSpecificActivityLocked(next, true, true);
    }
    ...
    return true;
} 

resumeTopActivityInnerLocked方法又调用了ActivityStackSupervisorstartSpecificActivityLocked方法:

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);

    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所在APP已启动
            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.
    }

    // 目标Activity所在APP未启动,通过zygote创建应用进程
    mService.startProcessLocked(r.processName, r.info.applicationInfo, true, 0,
            "activity", r.intent.getComponent(), false, false, true);
} 

startSpecificActivityLocked方法调用了realStartActivityLocked方法:

final boolean realStartActivityLocked(ActivityRecord r, ProcessRecord app,
            boolean andResume, boolean checkConfig) throws RemoteException {

    ...

    // 目标Activity所在APP已启动,通过此方法启动目标Activity
    app.thread.scheduleLaunchActivity(new Intent(r.intent), r.appToken,
        System.identityHashCode(r), r.info, new Configuration(mService.mConfiguration),
        new Configuration(task.mOverrideConfig), r.compat, r.launchedFromPackage,
        task.voiceInteractor, app.repProcState, r.icicle, r.persistentState, results,
        newIntents, !andResume, mService.isNextTransitionForward(), profilerInfo);

    ...

} 

上面这段代码中,其中app.thread的类型是IApplicationThreadIApplicationThread的声明如下:

public interface IApplicationThread extends IInterface {...} 

IApplicationThread继承了IInterface,所以它是一个Binder类型的接口,IApplicationThread内部包含了大量启动、停止Activity的接口,此外还包含启动和停止服务的接口。
IApplicationThread的实现者就是ActivityThread的内部类ApplicationThread,相关类的继承和实现关系如下所示:

private class ApplicationThread extends ApplicationThreadNative { ... }
public abstract class ApplicationThreadNative extends Binder implements IApplicationThread { ... } 

说了这么多,分析下来不仅累,还容易被绕晕,画张流程图总结下:

Android Activity生命周期,启动模式,启动过程详解

ApplicationThread

绕了一圈,Activity的启动过程还是回到了ApplicationThread中,ApplicationThread通过scheduleLaunchActivity方法启动Activity,源码如下:

public final void scheduleLaunchActivity(Intent intent, IBinder token, int ident,
        ActivityInfo info, Configuration curConfig, Configuration overrideConfig,
        CompatibilityInfo compatInfo, String referrer, IVoiceInteractor voiceInteractor,
        int procState, Bundle state, PersistableBundle persistentState,
        List<ResultInfo> pendingResults, List<ReferrerIntent> pendingNewIntents,
        boolean notResumed, boolean isForward, ProfilerInfo profilerInfo) {

    updateProcessState(procState, false);

    ActivityClientRecord r = new ActivityClientRecord();

    r.token = token;
    r.ident = ident;
    r.intent = intent;
    r.referrer = referrer;
    r.voiceInteractor = voiceInteractor;
    r.activityInfo = info;
    r.compatInfo = compatInfo;
    r.state = state;
    r.persistentState = persistentState;

    r.pendingResults = pendingResults;
    r.pendingIntents = pendingNewIntents;

    r.startsNotResumed = notResumed;
    r.isForward = isForward;

    r.profilerInfo = profilerInfo;

    r.overrideConfig = overrideConfig;
    updatePendingConfiguration(curConfig);

    sendMessage(H.LAUNCH_ACTIVITY, r);
} 

其主要实现就是发送一个启动消息交由Handler处理,这个Handler有个特殊的名字叫H(是不是很萌的名字),sendMessage的实现如下:

private void sendMessage(int what, Object obj, int arg1, int arg2, int seq) {
    if (DEBUG_MESSAGES) Slog.v(
            TAG, "SCHEDULE " + mH.codeToString(what) + " arg1=" + arg1 + " arg2=" + arg2 +
                    "seq= " + seq);
    Message msg = Message.obtain();
    msg.what = what;
    SomeArgs args = SomeArgs.obtain();
    args.arg1 = obj;
    args.argi1 = arg1;
    args.argi2 = arg2;
    args.argi3 = seq;
    msg.obj = args;
    mH.sendMessage(msg);
} 

H对Activity启动的消息进行处理:

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, "LAUNCH_ACTIVITY");
    Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
} 
break; 

ActivityThread

从上面代码可以看出,Activity的启动过程又转到ActivityThreadhandleLaunchActivity方法中,其源码如下:

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

    // 此方法中完成Activity生命周期的onCreate和onStart方法
    Activity a = performLaunchActivity(r, customIntent);

    if (a != null) {
        r.createdConfig = new Configuration(mConfiguration);
        reportSizeConfigurations(r);
        Bundle oldState = r.state;
        // 此方法中完成Activity生命周期的onResume方法
        handleResumeActivity(r.token, false, r.isForward,
                !r.activity.mFinished && !r.startsNotResumed, r.lastProcessedSeq, reason);

        ...
    }

    ...
} 

上述代码中的performLaunchActivity方法最终完成了Activity对象的创建和启动过程,performLaunchActivity这个方法主要完成了以下几件事:

摘自Android开发艺术探索

  1. 从ActivityClientRecord中获取待启动的Activity的组件信息
ActivityInfo aInfo = r.activityInfo;
if (r.packageInfo == null) {
    r.packageInfo = getPackageInfo(aInfo.applicationInfo, r.compatInfo,
            Context.CONTEXT_INCLUDE_CODE);
}

// ComponentName用来定义一个应用程序组件,例如Activity,Service,BroadcastReceiver,ContentProvider
ComponentName component = r.intent.getComponent();
if (component == null) {
    component = r.intent.resolveActivity(
        mInitialApplication.getPackageManager());
    r.intent.setComponent(component);
}

if (r.activityInfo.targetActivity != null) {
    component = new ComponentName(r.activityInfo.packageName,
            r.activityInfo.targetActivity);
} 
  1. 通过Instrumentation的newActivity方法使用类加载器创建Activity对象
Activity activity = null;
try {
    java.lang.ClassLoader cl = r.packageInfo.getClassLoader();
    // 通过类加载器来创建Activity对象
    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);
    }
} catch (Exception e) {
    if (!mInstrumentation.onException(activity, e)) {
        throw new RuntimeException(
            "Unable to instantiate activity " + component
            + ": " + e.toString(), e);
    }
} 
  1. 通过LoadedApk的makeApplication方法来创建Application对象
public Application makeApplication(boolean forceDefaultAppClass,
        Instrumentation instrumentation) {
    // 先判断Application是否为空,如果Application被创建过,就不会再重复创建,一个应用只有一个Application对象
    if (mApplication != null) {
        return mApplication;
    }

    Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "makeApplication");

    Application app = null;

    String appClass = mApplicationInfo.className;
    if (forceDefaultAppClass || (appClass == null)) {
        appClass = "android.app.Application";
    }

    try {
        java.lang.ClassLoader cl = getClassLoader();
        if (!mPackageName.equals("android")) {
            Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER,
                    "initializeJavaContextClassLoader");
            initializeJavaContextClassLoader();
            Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
        }
        ContextImpl appContext = ContextImpl.createAppContext(mActivityThread, this);
        // 通过类加载器创建Application对象
        app = mActivityThread.mInstrumentation.newApplication(
                cl, appClass, appContext);
        appContext.setOuterContext(app);
    } catch (Exception e) {
        if (!mActivityThread.mInstrumentation.onException(app, e)) {
            Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
            throw new RuntimeException(
                "Unable to instantiate application " + appClass
                + ": " + e.toString(), e);
        }
    }
    mActivityThread.mAllApplications.add(app);
    mApplication = app;

    if (instrumentation != null) {
        try {
            // 调用Application的onCreate方法
            instrumentation.callApplicationOnCreate(app);
        } catch (Exception e) {
            if (!instrumentation.onException(app, e)) {
                Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
                throw new RuntimeException(
                    "Unable to create application " + app.getClass().getName()
                    + ": " + e.toString(), e);
            }
        }
    }

    ...

    return app;
} 
  1. 创建ContextImpl对象,并通过Activity的attach方法来完成一些重要数据的初始化
Context appContext = createBaseContextForActivity(r, activity);
CharSequence title = r.activityInfo.loadLabel(appContext.getPackageManager());
Configuration config = new Configuration(mCompatConfiguration);
if (r.overrideConfig != null) {
    config.updateFrom(r.overrideConfig);
}
if (DEBUG_CONFIGURATION) Slog.v(TAG, "Launching activity "
        + r.activityInfo.name + " with config " + config);
Window window = null;
if (r.mPendingRemoveWindow != null && r.mPreserveWindow) {
    window = r.mPendingRemoveWindow;
    r.mPendingRemoveWindow = null;
    r.mPendingRemoveWindowManager = null;
}
activity.attach(appContext, this, getInstrumentation(), r.token,
    r.ident, app, r.intent, r.activityInfo, title, r.parent,
    r.embeddedID, r.lastNonConfigurationInstances, config,
    r.referrer, r.voiceInteractor, window); 

ContextImpl是一个很重要的数据结构,很明显是Context的具体实现,Context中大部分逻辑都是由ContextImpl实现的。ContextImpl通过Activity的attach方法来和Activity建立关联,同时在attach方法中还会完成Window的创建并建立自己和Window的关联,Window接收到外部事件输入后将事件传递给Activity。

  1. 调用Activity的onCreate方法
mInstrumentation.callActivityOnCreate(activity, r.state); 

上述代码中调用Activity的onCreate方法,意味着Activity已经完成了整个启动过程。此外还调用了Activity的onStartOnRestoreInstanceState方法:

activity.performStart();
...
mInstrumentation.callActivityOnRestoreInstanceState(activity, r.state); 

总结

至此,整个Activity的启动流程已经分析完了,其中还有很多细节需要打磨:

  • 启动过程中涉及到的类的作用分别是什么,我们并没有深入分析
  • 可以看出,Activity启动最后涉及到Handler消息循环机制,可以参考我的上一篇文章:Android消息机制Handler
  • Activity启动过程中AMS的地位显著,但是进程间通信的具体本质我也没有分析,下一步可以分析下Android IPC机制
  • Android从开机到点击Icon打开APP的过程中涉及到Launcher(它其实也是一个android应用程序)的启动以及APP进程的创建,我们此处没有提及,以后的博客中我也会整理分析
点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
刘望舒 刘望舒
2年前
Android深入四大组件(一)应用程序启动过程(前篇)
Android框架层Android深入四大组件categories:Android框架层本文首发于微信公众号「后厂技术官」前言在此前的文章中,我讲过了Android系统启动流程和Android应用进程启动过程,这一篇顺理成章来学习Android7.0的应用程序的启动过程。分析应用程序的启动过程其实就是分析根Activity的启动过程。<!more1
Wesley13 Wesley13
2年前
android Activity之间的参数传递
在开发web程序的时候,我们都知道web页面之间是可以传递参数的,在android中,每一个activity就相当于一个页面(窗口),当然,它也是可以传递参数的。思路:通过Intent激活组件(Activity),附带数据Intent(意向):一:通过Intent对象可以激活组件activity,具体有五种方式:方式1:
刘望舒 刘望舒
2年前
Android深入四大组件(二)Service的启动过程
Android框架层Android深入四大组件categories:Android框架层本文首发于微信公众号「刘望舒」前言此前我用较长的篇幅来介绍Android应用程序的启动过程(根Activity的启动过程),这一篇我们接着来分析Service的启动过程。建议阅读此篇文章前,请先阅读和这两篇文章。<!more1.ContextImpl到Activi
Wesley13 Wesley13
2年前
Activity启动模式
当应用运行起来后就会开启一条线程,线程中会运行一个任务栈,当Activity实例创建后就会放入任务栈中。Activity启动模式的设置在AndroidManifest.xml文件中,通过配置Activity的属性android:launchMode""设置。1\.Standard模式(默认)我们平时直接创建的Activity都是这种模式的
Stella981 Stella981
2年前
Android Activity生命周期与启动模式
Activity的完整生命周期如下图:!(http://static.oschina.net/uploads/img/201603/29145834_pCTs.gif)Activity的加载模式有四种:standard:标准模式,默认的加载模式,每次通过这种模式启动目标Acitivity,都创建一个新的实例,并将该Activity添加到当前
Stella981 Stella981
2年前
Android 面试常问七道题
1Android的四大组件以及作用Activity:Activity是Android程序与用户交互的窗口,是Android构造块中最基本的一种,它需要为保持各界面的状态,做很多持久化的事情,妥善管理生命周期以及一些跳转逻辑。service:后台服务于Activity,封装有一个完整的功能逻辑实现,接受上层指令,完成相关的动作,
Stella981 Stella981
2年前
Android蓝牙连接汽车OBD设备
//设备连接public class BluetoothConnect implements Runnable {    private static final UUID CONNECT_UUID  UUID.fromString("0000110100001000800000805F9B34FB");
Wesley13 Wesley13
2年前
Unity横屏
Android下发现Unity里面的Player设置,并不能完全有效,比如打开了自动旋转,启动的时候还是会横屏,修改XML添加以下代码<applicationandroid:icon"@drawable/ic\_launcher"                    android:label"@string/app\_name"
Easter79 Easter79
2年前
TextView中使用链接打开Activity
Activity加入intentfilter支持<activityandroid:name"TestActivity"<intentfilter<dataandroid:host"yourhost"android:scheme"testschema"/<
待兔
待兔
Lv1
男 · helloworld公司 · CTO - helloworld开发者社区站长
helloworld开发者社区网站站长
文章
89
粉丝
43
获赞
75