Framework - ActivityThread 应用启动UI渲染流程

一、概念

ActivityThread拥有 main(String[] agrs) 方法,作为程序的入口,是应用程序的初始化类。(ActivityThread不是主线程,它在 main() 方法中实例化,是运行在主线程中。)
ApplicationThread是 ActivityThread 的子类,用作 ActivityThread 与 AMS 进行 BInder 通信的桥梁。
Instrumentation管理 Application 和 Activity 声明周期的类,会在自己对应的方法中传入监听对象,执行该对象的生命周期方法。
ActivityStackActivityThread 中对 Activity 的管理栈,用来记录已经启动的Activity的先后关系、状态信息、是否需要启动新的进程。
ActivityRecord用来映射存储从 AMS 中获取的 Activity 信息的数据类。

二、流程

2.1 应用启动

  1. 点击桌面APP图标时会执行 Launcher.startActivity() ,通过 Binder 通信调用 system_server 进程中 AMS.startActivity() 发起启动请求。
  2. system_server 进程接收到请求后,向 Zygote 进程发送创建进程的请求。
  3. Zygote 进程 fork 出 APP 进程,并执行 ActivityThread.main() 创建主线程、初始化MainLooper、主线程Handler,同时初始化 ApplicationThread 用于和 AMS 通信。
  4. ActivityThread 传递 ApplicationThread 通过 Binder 通信向 AMS 中获取应用信息(即 APP 进程通过 Binder 调用 sytem_server 进程中 AMS 的 attachApplication() 方法)。
  5. AMS 对数据进行一些处理工作后,通过 Binder IPC 向 ActivityThread 发送handleBindApplication 请求(创建启动Application)和 scheduleLaunchActivity 请求(创建启动Activity)。
  6. 在处理 ApplicationThread 回调中返回的应用信息时,通过 Handler 向主线程发送 BIND_APPLICATION 和 LAUNCH_ACTIVITY 两条 Message(这里注意的是 AMS 和主线程并不直接通信,而是 AMS 和主线程的内部类 ApplicationThread 通过 Binder通信,ApplicationThread 再和主线程通过Handler消息交互)。
  7. 主线程在收到 Message 后,反射创建 Application 并执行 onCreate() 生命周期,再通过反射创建 Activity 并执行 onCreate() 生命周期。
  8. 到此 APP 便正式启动,开始进入一系列生命周期 onCreate/onStart/onResume,UI渲染后显示APP主界面。

2.2 UI渲染

2.2.1 onCreate()

2.2.2 onResume()

三、源码分析

3.1 入口 main()

由 Zygote 调用,进程创建后对其进行初始化让APP跑起来,ZygoteInit.zygoteInit() 通过反射找到 ActivityThread.main() 并调用。

public static void main(String[] args) {//主线程Looper的初始化Looper.prepareMainLooper();//实例化ActivityThreadActivityThread thread = new ActivityThread();//启动应用thread.attach(false);//主线程Handler初始化if (sMainThreadHandler == null) {sMainThreadHandler = thread.getHandler();}Looper.loop();//一旦上方的主线程loop()循环被终止便抛异常throw new RuntimeException("Main thread loop unexpectedly exited");
}

3.2 初始化主线程Looper

  • 在 Handler 源码中可以看出,初始化主线程 Looper 的参数 quitAllowed 为 false 表示不允许退出,该参数最终会在 MessageQueue 调用 quit() 时进行判断,如果是主线程会抛异常。
  • 主线程 Looper 初始化后会赋值给成员变量 sMainLooper,通过方法 getMainLooper() 向其它线程提供主线程的 Looper 对象。
    //主线程Looper的初始化public static void prepareMainLooper() {prepare(false);synchronized (Looper.class) {if (sMainLooper != null) {throw new IllegalStateException("The main Looper has already been prepared.");}sMainLooper = myLooper();}}//普通线程Looper的初始化public static void prepare() {prepare(true);}private static void prepare(boolean quitAllowed) {if (sThreadLocal.get() != null) {throw new RuntimeException("Only one Looper may be created per thread");}sThreadLocal.set(new Looper(quitAllowed));}
    public static Looper getMainLooper() {synchronized (Looper.class) {return sMainLooper;}}

3.3 应用启动过程

ActivityThread 向 AMS 中获取信息,AMS将信息封装成对象后回调过来。ActivityThread是逻辑的执行者,AMS是信息的持有者,他们之间通过Binder通信,ApplicationThread是两者通信的桥梁(回调)。

    //ActivityThread中private void attach(boolean system) {//获取系统服务AMS进行Binder通信if (!system) {final IActivityManager mgr = ActivityManager.getService();try {//传入用来通信的桥梁 ApplicationThreadmgr.attachApplication(mAppThread);} ...}}
    //AMS中public final void attachApplication(IApplicationThread thread) {synchronized (this) {attachApplicationLocked(thread, callingPid);}}private final boolean attachApplicationLocked(IApplicationThread thread, int pid) {//启动Applicationthread.bindApplication(...);//启动Activityif (mStackSupervisor.attachApplicationLocked(app)) {...}}

3.3.1 启动 Application

attachApplicationLocked() 中会将 Application 信息赋值给成员变量 ProcessRecord(是一个数据类用来存储 Application 的信息),通过调用 ApplicationThread.bindApplication() 将应用信息回传给 ActivityThread 进行处理。发送 Message 然后 Handler 处理(从Binder线程切换到主线程),调用重写的 handleBindApplication(),通过传过来的 AppBindData 的 makeApplication() 创建 Application 对象,具体是调用 Instrumentation 的newApplication(),通过类加载器类加载器反射加载Application类,执行 onCreate() 生命周期。

  • 如果未指定程序的 Application 则使用默认的 Application 类,否则使用我们指定的。
  • 在创建 Application 对象调用 attachBaseContext() 和 onCreate() 方法之间会调用 ContentProvider的onCreate() 方法这也是很多第三方SDK使用该特性实现初始化的原理。
    //ActivityThread中public final void bindApplication(...) {//对从AMS中获取的信息进行封装...//发送MessagesendMessage(H.BIND_APPLICATION, data);}//Handler处理会调用到这里来private void handleBindApplication(AppBindData data) {try {//创建ApplicationApplication app = data.info.makeApplication(data.restrictedBackupMode, null);mInitialApplication = app;//cotentprovider初始化,即为什么能用它初始化业务代码installContentProviders(app, data.providers); try {//启动生命周期onCreate()mInstrumentation.callApplicationOnCreate(app);}}
    //LoadedApk中public Application makeApplication(boolean forceDefaultAppClass, Instrumentation instrumentation) {//如果存在Application的实例,则直接返回,这也说明Application是个单例if (mApplication != null) {return mApplication;}Application app = null;//反射初始化Application...//交由 Instrumentation 创建并调用 Application 的 onCreate()if (instrumentation != null) {try {instrumentation.callApplicationOnCreate(app);}}return app;}

3.3.2 启动 Activity

attachApplicationLocked() 中会调用 stack.getAllRunningVisiableActivitiesLocked() 将栈中所有的 Activity 信息都赋值给一个列表 mTmpActivityList 然后遍历生成一个个配置类对象 ActivityRecord,通过 realStartActivityLocked() 传递,方法体中通过 ClientTransaction.obtain() 创建 Activity 启动事务并添加一个回调持有 LaunchActivityItem,事务被提交给 ClientLifecyleManager 最通过 ApplicationThread 回调到 ActivityThread 中处理。发送 Message 然后 Handler 处理(从Binder线程切换到主线程),会调用 TransactionExecutor 执行事务。 方法中会遍历所有添加的回调,拿到 LaunchActivityItem 父类 ClientTransctionItem 调用 execute(),创建一个配置类对象 ActivityRecord 赋值各种信息,回调给ActivityThread.handleLaunchActivity(),调用 performLaunchActivity(),进而通过 Instrumentation.newActivity() 创建 Activity(通过Intent得到类名用类加载器反射加载),通过 Instrumentation.callActivityOnCreate() 执行 onCreate() 生命周期。

    boolean attachApplicationLocked(ProcessRecord app) throws RemoteException {if (realStartActivityLocked(hr, app, true, true)) {...}          }final boolean realStartActivityLocked(ActivityRecord r, ProcessRecord app, boolean andResume, boolean checkConfig) throws RemoteException {try {调用ApplicationThread的scheduleLaunchActivity用于启动一个Activityapp.thread.scheduleLaunchActivity(...);}}
//ActivityThread中
private void handleLaunchActivity(ActivityClientRecord r, Intent customIntent, String reason) {//创建ActivityActivity a = performLaunchActivity(r, customIntent);//渲染Activityif (a != null) {handleResumeActivity(r.token, false, r.isForward, !r.activity.mFinished && !r.startsNotResumed, r.lastProcessedSeq, reason);}
}

3.4 UI 渲染过程

3.4.1 设置布局

Activity 被启动后会执行 onCreate(),我们会在其中调用 setContentView() 来设置布局文件,该方法会调用 getWindow() 获取自己持有的 Window 对象(子类PhoneWindow,创建时在 performLaunchActivit() 中调用 activity.attach() 时被赋值的)并调用它的 setContentView(),初始化 DecorView 并将我们自定义的布局加载到 content 中。

//Activity中
public void setContentView(@LayoutRes int layoutResID) {getWindow().setContentView(layoutResID);initWindowDecorActionBar();
}//PhoneWindow中
public void setContentView(int layoutResID) {if (mContentParent == null) {//初始化DecorViewinstallDecor();}
}
private void installDecor() {//初始化DecorView(加载系统的基础布局)mDecor = generateDecor(-1);//获取DecorView中Content部分,加载我们自定义的布局内容mContentParent = generateLayout(mDecor);
}

3.4.2 渲染布局

通过 addView() 将控件添加到 ViewGroup 中会触发 requestLayout() 和 invalidate(true)  重新布局和刷新,由于 DacorView 没有父容器所以只会执行添加操作。

//ViewGroup中
public void addView(View child, int index, LayoutParams params) {requestLayout();invalidate(true);
}//View中
public void requestLayout() {if (mParent != null && !mParent.isLayoutRequested()) {mParent.requestLayout();}
}

        对于 Activity 生命周期一直说 onStart() 是可见的,performStart() 中调用 Instrumentation.callActivityOnStart() 并没有做渲染处理,所以这个“可见”严谨来说是针对是否渲染。

        在 ActivityThread.handleResumeActivity() 中将 DecorView 添加到了 ViewManager 中,进而调用 WindowManagerGlobal.addView(),将 DecorView 设置给了在这里创建的 ViewRootImpl 对象,ViewRootImpl 会执行 requestLayout() → scheduleTraversals() → doTraversal() → performTraversals(),而这最后一个方法里会执行 View 的绘制流程(测量、摆放、绘制),此时 Activity 就被渲染出来了。

//ActivityThread中
final void handleResumeActivity(IBinder token, boolean clearHide, boolean isForward, boolean reallyResume, int seq, String reason) {//调用Activity.onResume()r = performResumeActivity(token, clearHide, reason);if (r != null) {final Activity a = r.activity;if (r.window == null && !a.mFinished && willBeVisible) {r.window = r.activity.getWindow();View decor = r.window.getDecorView();decor.setVisibility(View.INVISIBLE);ViewManager wm = a.getWindowManager();WindowManager.LayoutParams l = r.window.getAttributes();a.mDecor = decor;...//间接调用到WindowManagerGlobal的addView方法wm.addView(decor, l);}}
}
//WindowManagerGlobal中
public void addView(View view, ViewGroup.LayoutParams params, Display display, Window parentWindow) {root = new ViewRootImpl(view.getContext(), display);//Decor设置给了ViewRootImplroot.setView(view, wparams, panelParentView);
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/448698.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL 中 int(1) 和 int(10) 会影响存储的长度吗

一、MySQL 中 int(1) 和 int(10) 在MySQL数据库设计中,经常会遇到 int 类型的字段,并会习惯性的指定长度,比如: int(1) 和int(10),而一些新手可能会误解它们之间的关系,认为 int(10) 能够存储更多的数据。…

项目开发 多行编辑

问题 项目开发中,如何进行多行编辑 详细问题 笔者使用IDEA,Android Studio进行项目开发时,由于代码冗余,修改过程中若是逐一删除或编辑,效率相对低,如何进行多行删除或编辑 本文将提供IDEA,A…

【深度学习】基于PyTorch架构神经网络学习总结(基础概念基本网络搭建)

神经网络整体架构 类似于人体的神经元 神经网络工作原来为层次结构,一层一层的变换数据。如上述示例有4层,1层输入层、2层隐藏层、1层输出层神经元:数据的量或矩阵的大小,如上述示例中输入层中有三个神经元代表输入数据有3个特征…

C语言中的指针详解

大家好,今天给大家介绍C语言中的指针详解,文章末尾附有分享大家一个资料包,差不多150多G。里面学习内容、面经、项目都比较新也比较全!可进群免费领取。 **指针是C语言中的一个重要概念,它提供了一种直接访问内存地址…

某赛通电子文档安全管理系统 UploadFileToCatalog SQL注入漏洞复现

0x01 产品简介 某赛通电子文档安全管理系统(简称:CDG)是一款电子文档安全加密软件,该系统利用驱动层透明加密技术,通过对电子文档的加密保护,防止内部员工泄密和外部人员非法窃取企业核心重要数据资产,对电子文档进行全生命周期防护,系统具有透明加密、主动加密、智能…

Redis——事件

Redis服务器是一个事件驱动程序,服务器需要处理以下两种事件: 文件事件(file event):Redis服务器通过套接字与客户端(或者其他Redis服务器)进行连接,而文件事件就是服务器对套接字操作的抽象(linux下一切皆文件,返回的…

unity角色触摸转向

1、挂载脚本到角色的父物体A上 2 、以屏幕左边的触摸为移动,右边为转向操作 3、加载角色时,将角色的父物体设置为A,须将角色的位置和角度置0 using System; using System.Collections; using System.Collections.Generic; using UnityEngin…

【C++初阶】--入门基础(二)

目录 一.C输出与输入 二.缺省参数 1.概念 2.缺省参数分类 (1) 全缺省参数 (2)半缺省参数 三.函数重载 1.概念 2.C支持函数重载的原理--名字修饰 四.引用 1.概念 2.语法 3.引用的特性 (1)引用在定义时必须初始化 (2)引用时不能改变指向 (3)一个变量…

Faster-Whisper 实时识别电脑语音转文本

Faster-Whisper 实时识别电脑语音转文本 前言项目搭建环境安装Faster-Whisper下载模型编写测试代码运行测试代码实时转写脚本 参考 前言 以前做的智能对话软件接的Baidu API,想换成本地的,就搭一套Faster-Whisper吧。 下面是B站视频实时转写的截图 项…

Vite与Webpack打包内存溢出问题优雅处理方式

Vite与Webpack打包内存溢出问题处理 文章目录 Vite与Webpack打包内存溢出问题处理1. Vite1. 打包错误提示2. 命令行方式解决3. 配置环境变量方式解决1. 设置变量2. 配置系统的环境变量 2. Webpack1. 打包错误提示2. 命令行方式解决3. 配置环境变量方式解决1. 设置变量2. 配置系…

Cocos XR的WebBox实现流程

1. 正常3D场景下的webview 1.1 组件角色 Cocos Creator正常3D场景下只有在UI组件才支持webview,即作为下图中的UI Nodes(Canvas Node)的子节点,和3D组件是隔离开的,不能显示在3D空间中,UI Nodes(Canvas Node)是一个平面内的矩形…

计算机视觉-PCV包、Vlfeat库、Graphviz库的下载安装配置及问题解决(使用anaconda3 python 3.8.5)

目录 一、PCV包配置 二、Vlfeat配置 三、在PCV包的sift.py文件中对路径进行修改 四、以上步骤所需注意的错误 五、Graphviz配置 一、PCV包配置 1.下载PCV包,点开网址直接下载安装包(不用解压),下载之后将安装包放在任意目录位置https://codeload.github.com/Li-Shu14…