【Android 启动过程】Activity 启动源码分析 ( Activity -> AMS、主线程阶段 )

简介: 【Android 启动过程】Activity 启动源码分析 ( Activity -> AMS、主线程阶段 )

文章目录

一、Activity 启动源码分析 ( Activity -> AMS 阶段 )





一、Activity 启动源码分析 ( Activity -> AMS 阶段 )


调用 startActivity(new Intent()); 方法启动 Activity , 调用方法原型如下 :


public class Activity extends ContextThemeWrapper
        implements LayoutInflater.Factory2,
        Window.Callback, KeyEvent.Callback,
        OnCreateContextMenuListener, ComponentCallbacks2,
        Window.OnWindowDismissedCallback,
        AutofillManager.AutofillClient, ContentCaptureManager.ContentCaptureClient {
    /**
     * 与 {@link #startActivity(Intent, Bundle)} 方法相同 , 
     * Bundle options 参数设置为 null
     *
     * @param intent 启动的意图.
     *
     * @throws android.content.ActivityNotFoundException
     *
     * @see #startActivity(Intent, Bundle)
     * @see #startActivityForResult
     */
    @Override
    public void startActivity(Intent intent) {
        this.startActivity(intent, null);
    }
}


完整代码查看 frameworks/base/core/java/android/app/Activity.java 地址 ;



在 public void startActivity(Intent intent) 方法中 , 执行 this.startActivity(intent, null) 方法 , 方法原型如下 :


public class Activity extends ContextThemeWrapper
        implements LayoutInflater.Factory2,
        Window.Callback, KeyEvent.Callback,
        OnCreateContextMenuListener, ComponentCallbacks2,
        Window.OnWindowDismissedCallback,
        AutofillManager.AutofillClient, ContentCaptureManager.ContentCaptureClient {
    /**
     * 开展一项新活动。您将不会收到关于何时开始的任何信息
     * 活动退出。此实现将覆盖基本版本,
     * 提供关于
     * 执行启动的活动。因为这个额外的
     * 信息,{@link Intent#FLAG_ACTIVITY_NEW_TASK}启动标志不可用
     * 必修的;如果未指定,则新活动将添加到
     * 调用方的任务。
     *
     * <p>此方法引发{@link android.content.ActivityNotFoundException}
     * 如果没有找到运行给定意图的活动。
     *
     * @param intent 要启动的意图.
     * @param options 有关如何启动活动的其他选项.
     * 更多细节查看 {@link android.content.Context#startActivity(Intent, Bundle)}
     * Context.startActivity(Intent, Bundle)} .
     *
     * @throws android.content.ActivityNotFoundException
     *
     * @see #startActivity(Intent)
     * @see #startActivityForResult
     */
    @Override
    public void startActivity(Intent intent, @Nullable Bundle options) {
        if (mIntent != null && mIntent.hasExtra(AutofillManager.EXTRA_RESTORE_SESSION_TOKEN)
                && mIntent.hasExtra(AutofillManager.EXTRA_RESTORE_CROSS_ACTIVITY)) {
            if (TextUtils.equals(getPackageName(),
                    intent.resolveActivity(getPackageManager()).getPackageName())) {
                // Apply Autofill restore mechanism on the started activity by startActivity()
                final IBinder token =
                        mIntent.getIBinderExtra(AutofillManager.EXTRA_RESTORE_SESSION_TOKEN);
                // Remove restore ability from current activity
                mIntent.removeExtra(AutofillManager.EXTRA_RESTORE_SESSION_TOKEN);
                mIntent.removeExtra(AutofillManager.EXTRA_RESTORE_CROSS_ACTIVITY);
                // Put restore token
                intent.putExtra(AutofillManager.EXTRA_RESTORE_SESSION_TOKEN, token);
                intent.putExtra(AutofillManager.EXTRA_RESTORE_CROSS_ACTIVITY, true);
            }
        }
        // 如果有选项 , 执行 startActivityForResult 3 参数方法
        if (options != null) {
            startActivityForResult(intent, -1, options);
        } else {
          // 如果没有选项 , 执行 startActivityForResult 2 参数方法
            // Note we want to go through this call for compatibility with
            // applications that may have overridden the method.
            startActivityForResult(intent, -1);
        }
    }
}


完整代码查看 frameworks/base/core/java/android/app/Activity.java 地址 ;



public void startActivityForResult(@RequiresPermission Intent intent, int requestCode, @Nullable Bundle options) 函数原型 :
public class Activity extends ContextThemeWrapper
        implements LayoutInflater.Factory2,
        Window.Callback, KeyEvent.Callback,
        OnCreateContextMenuListener, ComponentCallbacks2,
        Window.OnWindowDismissedCallback,
        AutofillManager.AutofillClient, ContentCaptureManager.ContentCaptureClient {
    /**
     * 启动一项活动,在该活动完成时希望获得结果。
     * 当此活动退出时,您的
     * 将使用给定的请求代码调用onActivityResult()方法。
     * 使用否定的requestCode与调用
     * {@link#startActivity}(该活动不是作为子活动启动的)。
     *
     * <p>请注意,此方法仅应与Intent协议一起使用
     * 定义为返回结果的。在其他协议中(例如
     * {@link Intent#ACTION_MAIN}或{@link Intent#ACTION_VIEW}),您可以
     * 在你期望的时候没有得到结果。例如,如果您选择的活动
     * 如果启动使用{@link Intent#FLAG_ACTIVITY_NEW_TASK},则不会
     * 运行您的任务,您将立即收到取消结果。
     *
     * <p>作为特例,如果使用requestCode调用startActivityForResult()
     * >=0,在初始的onCreate(Bundle savedInstanceState)/onResume()过程中
     * 活动,则在显示结果之前不会显示窗口
     * 从已启动的活动返回。这是为了避免可见
     * 重定向到其他活动时闪烁。
     *
     * <p>此方法引发{@link android.content.ActivityNotFoundException}
     * 如果没有找到运行给定意图的活动。
     *
     * @param intent 要启动的意图.
     * @param requestCode 如果>=0,则当活动退出时,此代码将在onActivityResult()中返回.
     * @param options 有关如何启动活动的其他选项。
     * See {@link android.content.Context#startActivity(Intent, Bundle)}
     * Context.startActivity(Intent, Bundle)} for more details.
     *
     * @throws android.content.ActivityNotFoundException
     *
     * @see #startActivity
     */
    public void startActivityForResult(@RequiresPermission Intent intent, int requestCode,
            @Nullable Bundle options) {
        if (mParent == null) {
            options = transferSpringboardActivityOptions(options);
    // 调用 Instrumentation 方法 
    // Instrumentation 负责启动 Application 和 Activity 
    // 插件化时 , 此处是一个 Hook 点
            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) {
                // 如果此开始要求结果,我们可以避免
                // 在收到结果之前,活动是可见的。背景
                // 此代码在onCreate(Bundle savedInstanceState)或onResume()期间将保持
                // 在此期间隐藏活动,以避免闪烁。
                // 这只能在请求结果时执行,因为
                // 这保证了我们会在
                // 不管发生什么,活动都已完成。
                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);
            }
        }
    }
}


完整代码查看 frameworks/base/core/java/android/app/Activity.java ;



分析 mInstrumentation.execStartActivity( this, mMainThread.getApplicationThread(), mToken, this, intent, requestCode, options) 方法 , 注意传入的参数是 7 77 个参数 , 函数原型为 :


public class Instrumentation {
    /**
     * 执行应用程序发出的startActivity调用。默认值
     * 实现负责更新任何活动的{@link ActivityMonitor}
     * 对象并将此调用分派给系统活动管理器;你可以
     * 覆盖此选项以监视应用程序启动活动,以及
     * 修改它执行时发生的情况。
     * 
     * <p>此方法返回一个{@link ActivityResult}对象,您可以
     * 在拦截应用程序调用时使用,以避免执行启动
     * 活动操作,但仍返回应用程序运行的结果
     * 期望。为此,重写此方法以捕获对start的调用
     * 活动,以便返回包含结果的新ActivityResult
     * 您希望应用程序能够看到,并且不要调用超级用户
     * 班级。请注意,应用程序仅在以下情况下才期望结果:
     * <var>requestCode</var>is&amp;gt;=0
     * 
     * <p>此方法引发{@link android.content.ActivityNotFoundException}
     * 如果没有找到运行给定意图的活动。
     * 
     * @param who 从中启动活动的上下文的参数。
     * @param contextThread 活动所在上下文的主线程正在启动。
     * @param token 标识要启动的系统的内部令牌活动;可能为空。
     * @param target 哪个活动正在执行启动(并因此接收任何结果);如果未进行此调用,则可能为null从一项活动中。
     * @param intent 启动的实际意图。
     * @param requestCode 标识符;如果调用方不希望得到结果设置小于 0。
     * @param options 参数选项添加选项。
     * @return 要强制返回特定结果,请返回包含所需数据的ActivityResult对象;
     * 否则返回null。默认实现总是返回null。
     *
     * @throws android.content.ActivityNotFoundException
     *
     * @see Activity#startActivity(Intent)
     * @see Activity#startActivityForResult(Intent, int)
     * @see Activity#startActivityFromChild
     *
     * {@hide}
     */
    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);
                    ActivityResult result = null;
                    if (am.ignoreMatchingSpecificIntents()) {
                        result = am.onStartActivity(intent);
                    }
                    if (result != null) {
                        am.mHits++;
                        return result;
                    } else if (am.match(who, null, intent)) {
                        am.mHits++;
                        if (am.isBlocking()) {
                            return requestCode >= 0 ? am.getResult() : null;
                        }
                        break;
                    }
                }
            }
        }
        try {
            intent.migrateExtraStreamToClipData();
            intent.prepareToLeaveProcess(who);
    // 在此处调用 AMS , Binder 机制获取 AMS
            int result = ActivityManager.getService()
                    .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) {
            throw new RuntimeException("Failure from system", e);
        }
        return null;
    }
}


完整代码查看 frameworks/base/core/java/android/app/Instrumentation.java ;


目录
相关文章
|
24天前
|
Android开发
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
70 6
|
25天前
|
消息中间件 Android开发 索引
Android面试高频知识点(4) 详解Activity的启动流程
Android面试高频知识点(4) 详解Activity的启动流程
25 3
|
26天前
|
缓存 前端开发 Android开发
Android实战之如何截取Activity或者Fragment的内容?
本文首发于公众号“AntDream”,介绍了如何在Android中截取Activity或Fragment的屏幕内容并保存为图片。包括截取整个Activity、特定控件或区域的方法,以及处理包含RecyclerView的复杂情况。
18 3
|
25天前
|
Android开发
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
17 0
|
2月前
|
Android开发 开发者
Android面试之Activity启动流程简述
每个Android开发者都熟悉的Activity,但你是否了解它的启动流程呢?本文将带你深入了解。启动流程涉及四个关键角色:Launcher进程、SystemServer的AMS、应用程序的ActivityThread及Zygote进程。核心在于AMS与ActivityThread间的通信。文章详细解析了从Launcher启动Activity的过程,包括通过AIDL获取AMS、Zygote进程启动以及ActivityThread与AMS的通信机制。接着介绍了如何创建Application及Activity的具体步骤。整体流程清晰明了,帮助你更深入理解Activity的工作原理。
51 0
|
4天前
|
搜索推荐 Android开发 开发者
探索安卓开发中的自定义视图:打造个性化UI组件
【10月更文挑战第39天】在安卓开发的世界中,自定义视图是实现独特界面设计的关键。本文将引导你理解自定义视图的概念、创建流程,以及如何通过它们增强应用的用户体验。我们将从基础出发,逐步深入,最终让你能够自信地设计和实现专属的UI组件。
|
6天前
|
Android开发 Swift iOS开发
探索安卓与iOS开发的差异和挑战
【10月更文挑战第37天】在移动应用开发的广阔舞台上,安卓和iOS这两大操作系统扮演着主角。它们各自拥有独特的特性、优势以及面临的开发挑战。本文将深入探讨这两个平台在开发过程中的主要差异,从编程语言到用户界面设计,再到市场分布的不同影响,旨在为开发者提供一个全面的视角,帮助他们更好地理解并应对在不同平台上进行应用开发时可能遇到的难题和机遇。
|
8天前
|
XML 存储 Java
探索安卓开发之旅:从新手到专家
【10月更文挑战第35天】在数字化时代,安卓应用的开发成为了一个热门话题。本文旨在通过浅显易懂的语言,带领初学者了解安卓开发的基础知识,同时为有一定经验的开发者提供进阶技巧。我们将一起探讨如何从零开始构建第一个安卓应用,并逐步深入到性能优化和高级功能的实现。无论你是编程新手还是希望提升技能的开发者,这篇文章都将为你提供有价值的指导和灵感。
|
6天前
|
存储 API 开发工具
探索安卓开发:从基础到进阶
【10月更文挑战第37天】在这篇文章中,我们将一起探索安卓开发的奥秘。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的信息和建议。我们将从安卓开发的基础开始,逐步深入到更复杂的主题,如自定义组件、性能优化等。最后,我们将通过一个代码示例来展示如何实现一个简单的安卓应用。让我们一起开始吧!
|
7天前
|
存储 XML JSON
探索安卓开发:从新手到专家的旅程
【10月更文挑战第36天】在这篇文章中,我们将一起踏上一段激动人心的旅程,从零基础开始,逐步深入安卓开发的奥秘。无论你是编程新手,还是希望扩展技能的老手,这里都有适合你的知识宝藏等待发掘。通过实际的代码示例和深入浅出的解释,我们将解锁安卓开发的关键技能,让你能够构建自己的应用程序,甚至贡献于开源社区。准备好了吗?让我们开始吧!
19 2