startActivity启动流程的源码学习

2024-05-09 10:58

本文主要是介绍startActivity启动流程的源码学习,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

举例,当进程A调用startActivity方法,启动进程B,并打开B的Activity,这个过程是怎样的?以下是学习笔记,基于Android 9.0,在线源码查看:https://www.androidos.net.cn/android/9.0.0_r8/xref

  1. 进程A调用startActivity方法,本质上是通过binder通信,调用IActivityManager#startActivity方法
    • 如果该调用在Activity中,方法调用过程:
      ContextImpl#startActivity -> mInstrumentation#execStartActivity -> IActivityManager#startActivity
      备注:IActivityManager对象保存在进程A的单例IActivityManagerSingleton中,进程启动时查询ServiceManager#getService获得,具有缓存作用
       
  2. SystemServer进程的binder线程响应此请求,ActivityManagerService#startActivity方法被调用,过程如下(备注:过程较复杂,涉及Activity,ActivityStack等一连串操作):
    • ActivityManagerService#startActivity
    • ActivityStarter#startActivityMayWait,其中会调用PMS#resolveIntent方法,解析Intent获得ResolveInfo,并创建ActivityInfo等对象
      ResolveInfo rInfo = mSupervisor.resolveIntent((intent, resolvedType, userId, xxx);
    • ActivityStarter#startActivity,其中会创建即将打开的ActivityRecord等
      ActivityRecord r = new ActivityRecord(mService, callerApp, callingPid, callingUid, xxxx);
    • ActivityStack#resumeTopActivityUncheckedLocked
    • ActivityStackSupervisor#startSpecificActivityLocked,这里很重要,会根据目标进程名,查询进程是否已存在,如果存在,将执行realStartActivityLocked方法;如果不存在,将执行startProcessLocked,冷启动目标进程
    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);......if (app != null && app.thread != null) {.......realStartActivityLocked(r, app, andResume, checkConfig);return;}......mService.startProcessLocked(r.processName, r.info.applicationInfo, true, 0,
    
    • 假设这里要启动新进程,执行
      ActivityManagerService#startProcessLocked -> ActivityManagerService#startProcess -> Process#start -> ZygoteProcess#startViaZygote
      a. 利用Socket通信通知Zygote进程,孵化出进程B(判断是否启动成功:Socket返回结果中目标进程pid < 0,则启动失败,抛出ZygoteStartFailedEx异常)
      b. 进程B启动成功后,方法继续执行到ActivityManagerService#handleProcessStartedLocked,其中pid等信息填入事先创建好的ProcessRecord对象中,其对应每一个App进程,保存在数组mPidsSelfLocked中

      mPidsSelfLocked是AMS的成员变量,类型:SparseArray<E>,本质是利用2个数组存储,key和object各一个。此处pid为key,ProcessRecord对象为object。查询方法为二分法
       

  3. 目标进程B启动后,入口函数是ActivityThread#main(静态方法),过程如下:
    • 初始化环境,创建ActivityThread对象,其子对象中如:
      a. mH,类型:ActivityThread#H,作用:主线程的Handler,处理如BIND_APPLICATION,CREATE_SERVICE等几十个消息类型,消息来源是mAppThread
      b. mAppThread,类型:ActivityThread#ApplicationThread,作用:binder实体对象,用于接收AMS的binder调用,并将消息转发mH处理
      c. mInstrumentation,类型:Instrumentation,很重要的基础类,官方描述:用来监控系统与应用的所有交互,例如newApplication,callApplicationOnCreate,callActivityOnCreate等都是其子方法
      注:平时解bug时发现,一些App的插件化hook框架,会动态替换mInstrumentation对象,如达到启动Activity时替换目标页面等目的
    • 调用ActivityThread#attach方法 [重要],其中调用IActivityManager#attachApplication,与AMS进行binder通信,将进程的mAppThread传给AMS,类似进行注册
      mAppThread,类型:ApplicationThread extends IApplicationThread.Stub,本质为binder对象,作为服务端,将来接收AMS的binder调用
    • main方法最后调用Looper.loop,进程B的主线程进入循环处理消息模式

 

    public static void main(String[] args) {// 3.1(Binder对象,继承自IApplicationThread.Stub)Environment.initForCurrentUser();Looper.prepareMainLooper();// 3.2ActivityThread thread = new ActivityThread();thread.attach(false, startSeq);if (sMainThreadHandler == null) {sMainThreadHandler = thread.getHandler();}// 3.3Looper.loop();throw new RuntimeException("Main thread loop unexpectedly exited");}

 

  1. SystemServer进程,作为服务端,接收到进程B的attachApplication方法调用,过程如下:
    • 从前面提到的mPidsSelfLocked中提取app对象,类型:ProcessRecord,存入applicationThread等信息
    • 注册死亡回调,以便App进程挂掉后,AMS能及时得到通知,并清除App的相关信息
    • 发起binder通信,利用applicationThread对象,调用进程B的ApplicationThread#bindApplication方法,经消息转发,最终由进程B主线程的mH对象处理,消息类型:BIND_APPLICATION
    • 检查是否有页面要显示
      ActivityStackSupervisor#attachApplicationLocked ->ActivityStackSupervisor#realStartActivityLocked(如果进程B有位于栈顶的Activity需显示) -> ClientTransaction#schedule
      利用applicationThread对象,调用进程B的ApplicationThread#scheduleTransaction方法,经消息转发,最终由进程B主线程的mH对象处理,操控Activity生命周期等,消息类型:EXECUTE_TRANSACTION
    • 检查是否有服务要启动
      ActiveServices#attachApplicationLocked -> ActivityStackSupervisor#realStartServiceLocked
      利用applicationThread对象,调用进程B的ApplicationThread#scheduleCreateService方法,后续同上类似
    • 检查是否有广播消息要处理,不再详细描述
    • 检查是否有BackupAgent要创建,不再详细描述

 

    private final boolean attachApplicationLocked(IApplicationThread thread,int pid, int callingUid, long startSeq) {// 4.1ProcessRecord app;if (pid != MY_PID && pid >= 0) {app = mPidsSelfLocked.get(pid);// 4.2try {AppDeathRecipient adr = new AppDeathRecipient(app, pid, thread);thread.asBinder().linkToDeath(adr, 0);app.deathRecipient = adr;} catch (RemoteException e) {// ...}// 4.3thread.bindApplication(processName, appInfo, providers, xxx);// 4.4mStackSupervisor.attachApplicationLocked(app)) {didSomething = true;}// 4.5didSomething |= mServices.attachApplicationLocked(app, processName);// 4.6didSomething |= sendPendingBroadcastsLocked(app);// 4.7thread.scheduleCreateBackupAgent(mBackupTarget.appInfo, xxx);return true;}

 

总结:

至此经前4步,至少5次关键的跨进程通信,startActivity过程已基本完成。简单描述:

  • 进程A调用IActivityManager#startActivity方法,发起binder通信
  • SystemServer进程作为服务端,AMS会检查进程B是否存在,如无,发起Socket通信,通知Zgyote进程孵化出进程B
  • 进程B启动后,调用IActivityManager#attachApplication方法,发起binder通信
  • SystemServer进程作为服务端,AMS会保存进程B的相关信息,发起binder通信,调用进程B的bindApplication,scheduleTransaction等方法
  • 进程B依次完成Application#onCreate等初始化,并执行scheduleTransaction,初始化要显示的Activity,过程结束

 

这篇关于startActivity启动流程的源码学习的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/973216

相关文章

SpringBoot整合Flowable实现工作流的详细流程

《SpringBoot整合Flowable实现工作流的详细流程》Flowable是一个使用Java编写的轻量级业务流程引擎,Flowable流程引擎可用于部署BPMN2.0流程定义,创建这些流程定义的... 目录1、流程引擎介绍2、创建项目3、画流程图4、开发接口4.1 Java 类梳理4.2 查看流程图4

Nexus安装和启动的实现教程

《Nexus安装和启动的实现教程》:本文主要介绍Nexus安装和启动的实现教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、Nexus下载二、Nexus安装和启动三、关闭Nexus总结一、Nexus下载官方下载链接:DownloadWindows系统根

java Long 与long之间的转换流程

《javaLong与long之间的转换流程》Long类提供了一些方法,用于在long和其他数据类型(如String)之间进行转换,本文将详细介绍如何在Java中实现Long和long之间的转换,感... 目录概述流程步骤1:将long转换为Long对象步骤2:将Longhttp://www.cppcns.c

Java中实现线程的创建和启动的方法

《Java中实现线程的创建和启动的方法》在Java中,实现线程的创建和启动是两个不同但紧密相关的概念,理解为什么要启动线程(调用start()方法)而非直接调用run()方法,是掌握多线程编程的关键,... 目录1. 线程的生命周期2. start() vs run() 的本质区别3. 为什么必须通过 st

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

Oracle修改端口号之后无法启动的解决方案

《Oracle修改端口号之后无法启动的解决方案》Oracle数据库更改端口后出现监听器无法启动的问题确实较为常见,但并非必然发生,这一问题通常源于​​配置错误或环境冲突​​,而非端口修改本身,以下是系... 目录一、问题根源分析​​​二、保姆级解决方案​​​​步骤1:修正监听器配置文件 (listener.

MySQL版本问题导致项目无法启动问题的解决方案

《MySQL版本问题导致项目无法启动问题的解决方案》本文记录了一次因MySQL版本不一致导致项目启动失败的经历,详细解析了连接错误的原因,并提供了两种解决方案:调整连接字符串禁用SSL或统一MySQL... 目录本地项目启动报错报错原因:解决方案第一个:第二种:容器启动mysql的坑两种修改时区的方法:本地

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio

MySQL启动报错:InnoDB表空间丢失问题及解决方法

《MySQL启动报错:InnoDB表空间丢失问题及解决方法》在启动MySQL时,遇到了InnoDB:Tablespace5975wasnotfound,该错误表明MySQL在启动过程中无法找到指定的s... 目录mysql 启动报错:InnoDB 表空间丢失问题及解决方法错误分析解决方案1. 启用 inno

spring-gateway filters添加自定义过滤器实现流程分析(可插拔)

《spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔)》:本文主要介绍spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔),本文通过实例图... 目录需求背景需求拆解设计流程及作用域逻辑处理代码逻辑需求背景公司要求,通过公司网络代理访问的请求需要做请