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查看更多精彩文章!

目录
相关文章
|
5月前
|
缓存 移动开发 JavaScript
如何优化UniApp开发的App的启动速度?
如何优化UniApp开发的App的启动速度?
1028 139
|
5月前
|
移动开发 JavaScript 应用服务中间件
【06】优化完善落地页样式内容-精度优化-vue加vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
【06】优化完善落地页样式内容-精度优化-vue加vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
768 5
【06】优化完善落地页样式内容-精度优化-vue加vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
|
6月前
|
存储 消息中间件 人工智能
【08】AI辅助编程完整的安卓二次商业实战-修改消息聊天框背景色-触发聊天让程序异常终止bug牵涉更多聊天消息发送优化处理-优雅草卓伊凡
【08】AI辅助编程完整的安卓二次商业实战-修改消息聊天框背景色-触发聊天让程序异常终止bug牵涉更多聊天消息发送优化处理-优雅草卓伊凡
482 10
【08】AI辅助编程完整的安卓二次商业实战-修改消息聊天框背景色-触发聊天让程序异常终止bug牵涉更多聊天消息发送优化处理-优雅草卓伊凡
|
10月前
|
Android开发 数据安全/隐私保护 开发者
Android自定义view之模仿登录界面文本输入框(华为云APP)
本文介绍了一款自定义输入框的实现,包含静态效果、hint值浮动动画及功能扩展。通过组合多个控件完成界面布局,使用TranslateAnimation与AlphaAnimation实现hint文字上下浮动效果,支持密码加密解密显示、去除键盘回车空格输入、光标定位等功能。代码基于Android平台,提供完整源码与attrs配置,方便复用与定制。希望对开发者有所帮助。
205 0
|
6月前
|
存储 前端开发 安全
实现“永久登录”:针对蜻蜓Q系统的用户体验优化方案(前端uni-app+后端Laravel详解)-优雅草卓伊凡
实现“永久登录”:针对蜻蜓Q系统的用户体验优化方案(前端uni-app+后端Laravel详解)-优雅草卓伊凡
297 5
|
9月前
|
存储
《仿盒马》app开发技术分享--未完成订单列表展示逻辑优化(61)
上一节我们实现订单与优惠券的联合提交时,我去到订单列表页面查看生成的订单信息,发现现在的订单从信息展示到价格计算全都是有问题的。所以紧急的把对应的问题修改一下。
271 70
|
8月前
|
存储 Android开发 数据安全/隐私保护
Thanox安卓系统增加工具下载,管理、阻止、限制后台每个APP运行情况
Thanox是一款Android系统管理工具,专注于权限、后台启动及运行管理。支持应用冻结、系统优化、UI自定义和模块管理,基于Xposed框架开发,安全可靠且开源免费,兼容Android 6.0及以上版本。
1020 4
|
9月前
|
数据库
《仿盒马》app开发技术分享-- 优惠券逻辑优化(58)
我们已经实现了优惠券的领取和展示,现在已经趋近于一个完整的电商应用了,但是这时候问题又来了,我们领取完优惠券之后,我们的新用户优惠券模块依然存在,他并没有消失,既然我们是从云数据库中查询的数据,那么我们需要找到一个字段跟他对应起来,来实现新用户领券后关闭这个模块的展示,同时我们在未登录的时候他也要保持隐藏,登录后能实现优惠券的领取。然后在结算的时候得出有几张符合的券能用
141 9
|
10月前
|
缓存 编解码 Android开发
Android内存优化之图片优化
本文主要探讨Android开发中的图片优化问题,包括图片优化的重要性、OOM错误的成因及解决方法、Android支持的图片格式及其特点。同时介绍了图片储存优化的三种方式:尺寸优化、质量压缩和内存重用,并详细讲解了相关的实现方法与属性。此外,还分析了图片加载优化策略,如异步加载、缓存机制、懒加载等,并结合多级缓存流程提升性能。最后对比了几大主流图片加载框架(Universal ImageLoader、Picasso、Glide、Fresco)的特点与适用场景,重点推荐Fresco在处理大图、动图时的优异表现。这些内容为开发者提供了全面的图片优化解决方案。
415 1
|
9月前
《仿盒马》app开发技术分享-- 逻辑优化第三弹(83)
现在我们的app功能已经趋近完善,bug和缺失的细节也越来越少了,我们继续对app进行优化,首先是我们的积分页面,我们只实现了全部的积分展示内容,对收入和支出的积分明细并没有进行展示,这里我们要实现一下,然后就是我们的优惠券,我们已过期的优惠券并没有修改状态为已过期。
152 0

热门文章

最新文章