Android应用启动流程之从启动到可交互的过程解析

 更新时间:2023年08月28日 09:12:48   作者:Android西红柿  
这篇文章将给大家总结学习Android 基础知识,Android应用启动流程,从启动到可交互的过程解析,在学习过程中,大家最好是把源码下载下来,感兴趣的小伙伴跟着小编一起来看看吧

一、导读

我们继续总结学习Android 基础知识,温故知新。

还是在多年前做系统应用开发的时候,通过编译源码学习了一遍APP的启动过程,
最近在整理应用启动优化相关的文档,打算重新梳理一遍,并整理成一个通俗易懂的文档,以备后面查看。
在学习过程中,大家最好是把源码下载下来。

1.1 启动知识储备

启动过程比较复杂,涉及到的知识点非常多,建议通过其他文章进行学习。

1.2 Zygote进程

Zygote进程是Android设备启动时由init进程创建和启动,之后不断接收来自应用程序的请求。
Zygote进程预先创建一个虚拟机和一些共享资源,以提高应用程序的启动速度和系统资源的利用效率。
同时,Zygote进程也负责处理应用程序进程的环境初始化和加载核心类库等操作。
每当有新的应用程序启动请求到达时,Zygote进程会fork出一个子进程(应用程序进程)来运行该应用程序。

1.3 SystemServer进程

SystemServer进程是Android操作系统的一个关键进程,它在设备启动时由init进程创建的。

SystemServer进程是一个守护进程,在系统运行期间一直存在,负责启动和管理各个核心系统服务,如AMS、Window Manager、Package Manager等。

1.4 AMS(Activity Manager Service)

AMS(Activity Manager Service)是SystemServer进程中的一个关键组件,负责管理所有应用程序的生命周期和交互。它监控应用程序的启动、暂停、恢复和销毁,
并协调应用程序之间的切换和交互。除此之外,AMS还处理任务栈管理、权限管理、用户通知等相关任务。

1.5 APP进程(ActivityThread)

APP进程是指在Android系统中运行的应用程序进程。每个运行的应用程序都会分配一个或多个独立的APP进程来承载应用的执行环境。
每个APP进程都是Zygote进程通过fork系统调用创建的子进程。

ActivityThread 是Android应用程序的主线程,负责管理和调度应用程序中的所有Activity的生命周期以及处理一些系统事件。

ActivityThread的主要职责包括:

  1. 创建和启动应用程序的主Activity。
  2. 处理应用程序的消息循环,接收和分发来自系统和应用程序的消息。
  3. 管理应用程序的所有Activity的生命周期,包括创建、启动、暂停、恢复、停止和销毁等。
  4. 处理应用程序的资源加载和更新,包括布局文件、图片、字符串等资源的加载。
  5. 处理应用程序的窗口管理,包括窗口的创建、显示、隐藏和更新等操作。
  6. 与AMS (Activity Manager Service)通信,处理Activity的启动、停止、切换等请求,并接收来自AMS的指令和请求。
  7. 处理应用程序的异常错误,包括崩溃、ANR (Application Not Responding)等情况的处理。

二、启动流程概览

涉及到多个进程之间的通信:
1、用户点击图标后,Laucher(手机桌面)会通过binder的方式通知SystemServer进程。
2、SystemServer收到信息,内部的AMS(Activity Manager Service)通知Zygote去创建APP进程,通过反射调用ActivityThread.main方法,即创建主线程。
3、APP进程创建后(Zygote进行fork),通过回调通知AMS。
4、AMS收到信息后经过一些列调用,通知APP进程(ActivityThread)创建Application
5、Application 之后,继续执行调用到ClientLifecycleManager,进行Activity创建等过程

我们画个简单的图

下面贴一下代码:

2.1 用户点击桌面图标

Launcher获取到AMS的binder,之后进入到SystemServer进程。

然后调用的zygote进程,fork出一个新的app进程。

2.2 创建app进程

ActivityThread

应用进程启动的入口函数main()方法中,会调用Looper相关的初始化和循环方法,从而让主线程的Looper进入循环状态,保证app进程的一直运行状态
    public static void main(String[] args) {
        . ..
        //looper绑定主线程
        Looper.prepareMainLooper();
        创建ActivityThread
        ActivityThread thread = new ActivityThread();
        thread.attach(false, startSeq);
        //looper开启无限循环读取消息
        Looper.loop();
            . ..
    }
    private void attach(boolean system, long startSeq) {
        if (!system) {
            . ..
            final IActivityManager mgr = ActivityManager.getService();
            try {
                // 通知 AMS, 并将当前进程的binder句柄传给AMS绑定管理起来
                mgr.attachApplication(mAppThread, startSeq);
            } catch (RemoteException ex) {
            }
            . ..
        }
        . ..
    }

attachApplication 这个流程会分两步:
1、创建应用的application对象
2、activity启动显示

2.3 创建 application

IActivityManager.aidl

    void attachApplication(in IApplicationThread app, long startSeq);

ActivityManagerService

// 这里接受APP传递过来的消息
    @Override
    public final void attachApplication(IApplicationThread thread, long startSeq) {
        synchronized (this) {
            // 通过 Binder 获取 PID ,PID和UID都是进程创建应用的时候系统指定的
            int callingPid = Binder.getCallingPid();
            final int callingUid = Binder.getCallingUid();
            final long origId = Binder.clearCallingIdentity();
            attachApplicationLocked(thread, callingPid, callingUid, startSeq);
        }
    }
    @GuardedBy("this")
    private final boolean attachApplicationLocked(IApplicationThread thread,
            int pid, int callingUid, long startSeq) {
        // *** 1、通知APP进程进行初始化操作,初始化application,创建应用进程的application对象
        if (app.isolatedEntryPoint != null) {
        } else if (app.instr != null) {
            thread.bindApplication(processName, appInfo, providers, ...);
        } else {
            thread.bindApplication(processName, appInfo, providers, ...);
        }
        // *** 2、 通知APP拉起指定的MainActivity,调用ATMS的启动页面的方法,后面继续介绍
        // See if the top visible activity is waiting to run in this process...
        if (normalMode) {
            try {
                if (mStackSupervisor.attachApplicationLocked(app)) {
                    didSomething = true;
                }
            } catch (Exception e) {
            }
        }
    }

IApplicationThread.aidl

void bindApplication(in String packageName, in ApplicationInfo info, ...);

我们来看看 Application 是如何创建的
ActivityThread

由AMS调用过来,具体看上面的代码
public final void bindApplication(String processName, ApplicationInfo appInfo, ...) {
        sendMessage(H.BIND_APPLICATION, data);
}
public void handleMessage(Message msg) {
    switch (msg.what) {
        case BIND_APPLICATION:
        AppBindData data = (AppBindData)msg.obj;
        handleBindApplication(data);
        break;
        . ..
    }
}
private void handleBindApplication(AppBindData data) {
    Application app;
    try{
        classLoader加载APK中的dex,并且加载APK的资源
        final LoadedApk pi = getPackageInfo(instrApp, data.compatInfo,
                appContext.getClassLoader(), false, true, false);
        final ContextImpl instrContext = ContextImpl.createAppContext(this, pi);
        通过 calssLoader 来创建 application 对象
        app=data.info.makeApplication(data.restrictedBackupMode,null);
        try{
            调用Application的onCreate
            mInstrumentation.onCreate(data.instrumentationArgs);
        }catch(Exception e){
        }
        try{
            mInstrumentation.callApplicationOnCreate(app);
        }catch(Exception e){
        }
    } ... 
}
public Application newApplication(ClassLoader cl, String className, Context context) 。。{
    Application app = getFactory(context.getPackageName()).instantiateApplication(cl, className);
    app.attach(context);
    return app;
}
(Application) cl.loadClass(className).newInstance();

2.4 activity启动显示

接着上面的代码 mStackSupervisor.attachApplicationLocked(app);

ActivityStackSupervisor
方法realStartActivityLocked();

    boolean attachApplicationLocked(ProcessRecord app) throws RemoteException {
        final String processName = app.processName;
        boolean didSomething = false;
        Activity启动的入口
        try {
            if (realStartActivityLocked(activity, app, top == activity, true)) {
            }
        } catch (RemoteException e) {
        }
        return didSomething;
    }
    真的开始启动 activity
    final boolean realStartActivityLocked(ActivityRecord r, ProcessRecord app,
            boolean andResume, boolean checkConfig) throws RemoteException {
            // Create activity launch transaction.
            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));
            // Schedule transaction.
            mService.getLifecycleManager().scheduleTransaction(clientTransaction);
    }

ActivityManagerService

    ClientLifecycleManager getLifecycleManager() {
        return mLifecycleManager;
    }

ClientLifecycleManager

    通过代码,我们可以看到,获取的client就是 ActivityThread,IApplicationThread是一个AIDL文件
    void scheduleTransaction(ClientTransaction transaction) throws RemoteException {
        final IApplicationThread client = transaction.getClient();
        transaction.schedule();
    }

ClientTransaction

        
    /** Target client. */
    private IApplicationThread mClient;
    public void schedule() throws RemoteException {
        mClient.scheduleTransaction(this);
    }

ClientTransactionHandler

    //ActivityThread中没有复写scheduleTransaction,会执行到父类的方法
    //public final class ActivityThread extends ClientTransactionHandler
    //ClientTransactionHandler.java
    public abstract class ClientTransactionHandler {
        void scheduleTransaction(ClientTransaction transaction) {
            transaction.preExecute(this);
            //发送消息
            sendMessage(ActivityThread.H.EXECUTE_TRANSACTION, transaction);
        }
    }

ActivityThread.java

    case EXECUTE_TRANSACTION:
        final ClientTransaction transaction = (ClientTransaction) msg.obj;
        mTransactionExecutor.execute(transaction);
        break;

继续一系列调用

最后调用到ActivityThread,创建并启动Activity

    /**
     * Extended implementation of activity launch. Used when server requests a launch or relaunch.
     */
    @Override
    public Activity handleLaunchActivity(ActivityClientRecord r,
            PendingTransactionActions pendingActions, Intent customIntent) {
        final Activity a = performLaunchActivity(r, customIntent);
        return a;
    }
    Activity实例化过程
    /**  Core implementation of activity launch. */
    private Activity performLaunchActivity(ActivityClientRecord r, Intent customIntent) {
        Activity activity = null;
        try {
            // 通过反射实例化Activity对象
            java.lang.ClassLoader cl = appContext.getClassLoader();
            activity = mInstrumentation.newActivity(cl, component.getClassName(), r.intent);
        } catch (Exception e) {
        }
        window
        theme
        //当实例化Activity对象后,继续执行callActivityOnCreate
        if (r.isPersistable()) {
            mInstrumentation.callActivityOnCreate(activity, r.state, r.persistentState);
        } else {
            mInstrumentation.callActivityOnCreate(activity, r.state);
        }
    }

接下来就是执行 Activity 生命周期函数

所用的源码不同,代码可能会有些许差别。

以上就是Android应用启动流程之从启动到可交互的过程解析的详细内容,更多关于Android应用启动流程解析的资料请关注脚本之家其它相关文章!

相关文章

  • Android Studio下的APP目录结构详解

    Android Studio下的APP目录结构详解

    这篇文章主要介绍了AndroidStudio下的APP目录结构,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-05-05
  • Android编写Router路由框架实例过程详解

    Android编写Router路由框架实例过程详解

    为什么要用路由框架,路由框架哪些好处等等,在此就不做解释
    最常用的框架是ARouter,那是不是可以自己写一个路由框架呢,不参考ARouter的方式
    2023-04-04
  • Kotlin Flow常见场景下的使用实例

    Kotlin Flow常见场景下的使用实例

    这篇文章主要为大家介绍了Kotlin Flow常见场景下的使用实例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • 如何安装adb工具及常用的adb命令

    如何安装adb工具及常用的adb命令

    ADB全称为Android Debug Bridge,起到调试桥的作用,是一个客户端-服务器端程序,ADB 也是 Android SDK 中的一个工具,可以直接操作管理 Android 模拟器或者真实的 Android 设备,本文介绍如何安装adb工具及常用的adb命令,感兴趣的朋友一起看看吧
    2024-01-01
  • Android 添加TextView删除线(代码简单)

    Android 添加TextView删除线(代码简单)

    最近接了个项目,其中有项目需求是这样的,有这么个需求,就是一个产品下有两个价格,一个是市场价,一个是销售价,这时要把市场价添加个删除线;怎么实现呢?下面小编给大家分享一段简单的代码实现Android 添加TextView删除线
    2016-02-02
  • Android实现底部缓慢弹出菜单

    Android实现底部缓慢弹出菜单

    这篇文章主要介绍了Android实现底部缓慢弹出菜单,从底部缓慢弹出一个Button Menu,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10
  • Android数据库增删改查实战案例

    Android数据库增删改查实战案例

    我们在编程中经常会遇到数据库的操作,这篇文章主要给大家介绍了关于Android数据库增删改查的相关资料,文中通过图文以及实例代码介绍的非常详细,需要的朋友可以参考下
    2023-04-04
  • Android中GPS坐标转换为高德地图坐标详解

    Android中GPS坐标转换为高德地图坐标详解

    最近因为公司需求,在做GPS定位,并且将获得的坐标显示在高德地图上,但是实际效果跟我们期望的是有偏差的。通过查阅资料,才知道有地球坐标、火星坐标之说。下面这篇文章就详细介绍了Android中GPS坐标转换为高德地图坐标的方法,需要的朋友可以参考下。
    2017-01-01
  • Android实现短信、微信、微博分享功能

    Android实现短信、微信、微博分享功能

    微信、微博分享功能大家都体验过吧,非常方便我们的生活,下面通过本文给大家介绍Android实现短信、微信、微博分享功能,需要的朋友参考下吧
    2017-12-12
  • flutter实现底部不规则导航栏

    flutter实现底部不规则导航栏

    这篇文章主要为大家详细介绍了flutter实现底部不规则导航栏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-07-07

最新评论