Android面试题之App的启动流程和启动速度优化

简介: App启动流程概括:当用户点击App图标,Launcher通过Binder IPC请求system_server启动Activity。system_server指示Zygote fork新进程,接着App进程向system_server申请启动Activity。经过Binder通信,Activity创建并回调生命周期方法。启动状态分为冷启动、温启动和热启动,其中冷启动耗时最长。优化技巧包括异步初始化、避免主线程I/O、类加载优化和简化布局。

本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点

App启动流程

①点击桌面App图标,Launcher进程采用Binder IPC向system_server进程发起startActivity请求;

②system_server进程接收到请求后,向zygote进程发送创建进程的请求;

③Zygote进程fork出新的子进程,即App进程;

④App进程,通过Binder IPC向sytem_server进程发起attachApplication请求;

⑤system_server进程在收到请求后,进行一系列准备工作后,再通过binder IPC向App进程发送scheduleLaunchActivity请求;

⑥App进程的binder线程(ApplicationThread)在收到请求后,通过handler向主线程发送LAUNCH_ACTIVITY消息;

⑦主线程在收到Message后,通过反射机制创建目标Activity,并回调Activity.onCreate()等方法。

⑧到此,App便正式启动,开始进入Activity生命周期,执行完onCreate/onStart/onResume方法,UI渲染结束后便可以看到App的主界面。

启动状态

应用有三种启动状态,每种状态都会影响应用向用户显示所需的时间:冷启动、温启动与热启动。

  • 温启动包含了在冷启动期间发生的部分操作;同时,它的开销要比热启动高
  • 冷启动是指应用从头开始启动:系统进程在冷启动后才创建应用进程
  • 在热启动中,系统的所有工作就是将 Activity 带到前台
  • 启动优化一般是优化冷启动,一般冷启动时间大于5s就认为时间过长(官方)

启动时间查看

  • 在logcat里,通过Display字段过滤,可以看到系统打印的启动时间日志
  • 用adb查看App启动时间
 adb shell am start -S -W [packageName]/[activityName]

一般会输入三个值:ThisTime、TotalTime与WaitTime。

一般关注TotalTime表示新应用启动的耗时,包括新进程的启动和Activity的启动,但不包括前一个应用Activity pause的耗时。

这个是统计到用户可操作的时间,也就是到onWindowFocusChanged的时间。

用IDE的CPU Profile功能来查看哪些步骤启动耗时
  • 首先需要打开AS中的CPU Profile开关,在App的run configuration设置里,找到Profile,
然后打开
start this recording on startup,
选择
trace java methods

之后重启App,AS就会自动打开Profile

  • 这个功能只支持API26以上的版本
  • Profile页面有一下4个tab
Call Chart 根据时间线查看调用栈,便于观察每次调用是何时发生的
Flame Chart 根据耗时百分比查看调用栈,便于发现总耗时很长的调用链
Top Down Tree 查看记录数据中所有方法调用栈,便于观察其中每一步所消耗的精确时间。
Bottom Up Tree 相对于Top Down Tree,能够更方便查看耗时方法如何被调用

此外,还可以用StrictMode模式检测耗时操作

 //debug下开启StrictMode,检查耗时操作
if (BuildConfig.DEBUG){
   
    //检查线程
    StrictMode.setThreadPolicy(StrictMode.ThreadPolicy.Builder()
        .detectDiskReads() //读写操作
        .detectDiskWrites()
        .detectNetwork() //检测网络
        .penaltyLog() //检测到以后输出日志
        .build())
    //检查虚拟机
    StrictMode.setVmPolicy(StrictMode.VmPolicy.Builder()
        .detectLeakedSqlLiteObjects() //SQL泄漏操作
        .detectLeakedClosableObjects() //未关闭的Closable
        .penaltyLog() //检测到以后输出日志
        .penaltyDeath() //违规奔溃
        .build())
}
第三方框架启动耗时
  • EventBus、Arouter等
黑白屏问题优化

通过给启动的Activity设置一个主题style,这个主题设置windowbackground为一张启动图,可以优化体验;然后再activity的oncreate中把主题设置回来

对于API26以下的版本,我们可以通过手动触发CPU Profile
  1. 首先在application中启动,test表示生成的.trace文件名称
Debug.startMethodTracing("test")
  1. 然后在启动的入口activity的onWindowFocusChanged方法中停止
Debug.stopMethodTracing()

重新启动App以后会在SD卡上生成test.trace文件(App需要有读写SD卡权限),直接双击就可以在AS中打开了

总结

1). 合理的使用异步初始化、延迟初始化、懒加载机制。

2). 启动过程避免耗时操作,如数据库 I/O操作不要放在主线程执行。

3). 类加载优化:提前异步执行类加载。

4). 合理使用IdleHandler进行延迟初始化。

  • IdleHandler会在MessageQueue队列消息处理完以后空闲了回调
  • 比如如果要在activity的View绘制完成之后再做些操作,就可以用IdleHandler;
  • onResume时View还没有绘制完成,onResume回调以后才会去执行performTraversals函数,也就是三大绘制流程
Looper.myQueue().addIdleHandler(new MessageQueue.IdleHandler() {
   
    @Override
    public boolean queueIdle() {
   
        long time = System.currentTimeMillis();
        //耗时操作
        Log.i("tag", "cost time " + (System.currentTimeMillis() - time));
        //false只执行一次,true会执行多次
        return false;
    }
});

5). 简化布局:

  • 简化非必要的层级
  • 注意一些自定义View里面是否有读写IO的情况,比如读写一些配置文件,字体文件等

欢迎关注我的公众号AntDream查看更多精彩文章!

目录
相关文章
|
6天前
|
Android开发
Android面试题之Activity的启动模式和flag
Android Activity的四种启动模式:standard(默认,每次启动创建新实例),singleTop(栈顶复用,不走onCreate,调用onNewIntent),singleTask(栈内唯一,清除上方Activity)和singleInstance(单独栈内唯一)。启动模式在AndroidManifest.xml中配置,Intent Flags如FLAG_ACTIVITY_CLEAR_TOP和FLAG_ACTIVITY_SINGLE_TOP可实现类似功能。了解这些对于处理Activity栈管理至关重要。
12 0
|
3天前
|
Android开发 Kotlin
Android面试题之kotlin中怎么限制一个函数参数的取值范围和取值类型等
在Kotlin中,限制函数参数可通过类型系统、泛型、条件检查、数据类、密封类和注解实现。例如,使用枚举限制参数为特定值,泛型约束确保参数为Number子类,条件检查如`require`确保参数在特定范围内,数据类封装可添加验证,密封类限制为一组预定义值,注解结合第三方库如Bean Validation进行校验。
15 6
|
3天前
|
Android开发
Android面试题之自定义View注意事项
在Android开发中,自定义View主要分为四类:直接继承View重写onDraw,继承ViewGroup创建布局,扩展特定View如TextView,以及继承特定ViewGroup。实现时需注意:支持wrap_content通过onMeasure处理,支持padding需在onDraw或onMeasure/onLayout中处理。避免在View中使用Handler,使用post系列方法代替。记得在onDetachedFromWindow时停止线程和动画以防止内存泄漏。处理滑动嵌套时解决滑动冲突,并避免在onDraw中大量创建临时对象。
13 4
|
23小时前
|
Android开发
Android面试题之View的invalidate方法和postInvalidate方法有什么区别
本文探讨了Android自定义View中`invalidate()`和`postInvalidate()`的区别。`invalidate()`在UI线程中刷新View,而`postInvalidate()`用于非UI线程,通过消息机制切换到UI线程执行`invalidate()`。源码分析显示,`postInvalidate()`最终调用`ViewRootImpl`的`dispatchInvalidateDelayed`,通过Handler发送消息到UI线程执行刷新。
11 1
|
4天前
|
Android开发 Kotlin
Android面试题之 Kotlin中退出迭代器的方式有哪些
在Android和Kotlin中,遍历集合时可使用迭代器结合`break`提前终止循环。例如,使用`while`和迭代器,或用`forEach`配合`return@forEach`来中断遍历。若需退出外层函数,可定义自定义标签。在遍历并删除元素时,这些技巧尤其有用。
11 3
|
10天前
|
存储 安全 Java
Android面试题之ArrayList源码详解
ArrayList是Java中基于数组实现的列表,提供O(1)的索引访问,但插入和删除操作平均时间复杂度为O(n)。默认容量为10,当需要时会通过System.arraycopy扩容。允许存储null,非线程安全。面试常问:List是接口,ArrayList是其实现之一,推荐使用List接口编程以实现更好的灵活性。更多详情见[ArrayList源码](http://grepcode.com/file/repository.grepcode.com/java/root/jdk/openjdk/8u40-b25/java/util/ArrayList.java#ArrayList.Node)。
11 2
|
11天前
|
Android开发
Android面试题经典之如何全局替换App的字体
在Android应用中替换字体有全局和局部方法。全局替换涉及在`Application`的`onCreate`中设置自定义字体,并创建新主题。局部替换则可在布局中通过`ResourcesCompat.getFont()`加载字体文件并应用于`TextView`。
21 2
|
4天前
|
Java 数据格式
Java面试题:简述Java Socket编程的基本流程,包括客户端和服务器的创建与通信。
Java面试题:简述Java Socket编程的基本流程,包括客户端和服务器的创建与通信。
11 0
|
5天前
|
存储 安全 Java
Java面试题:如何在Java应用中实现有效的内存优化?在多线程环境下,如何确保数据的线程安全?如何设计并实现一个基于ExecutorService的任务处理流程?
Java面试题:如何在Java应用中实现有效的内存优化?在多线程环境下,如何确保数据的线程安全?如何设计并实现一个基于ExecutorService的任务处理流程?
10 0
|
6天前
|
消息中间件 前端开发 Android开发
Android面试题自定义View之Window、ViewRootImpl和View的三大流程
Android开发中,View的三大核心流程包括measure(测量)、layout(布局)和draw(绘制)。MeasureSpec类在测量过程中起到关键作用,它结合尺寸大小和模式(EXACTLY、AT_MOST、UNSPECIFIED)来指定View应如何测量。onMeasure方法用于自定义View的测量,布局阶段,ViewGroup调用onLayout确定子元素位置,而draw阶段按照特定顺序绘制背景、内容、子元素和装饰。整个流程始于ViewRootImpl的performTraversals,该方法触发测量、布局和绘制。
13 0