每天一个面试知识点,日后拿下大厂面试官!
App启动流程三问
App的启动流程和Activity的启动流程也是面试中常常被问到的知识点,但是往往会涉及到很多方面,让我们不知道从何做答,今天就一起来看看启动流程相关问题:
- 说一下APP的启动流程
- 里面涉及到哪些重要的成员呢?都分别负责了什么,比如AMS?
- 启动流程了解后,对我们开发有什么帮助呢?
说一下APP的启动流程
Launcher
被调用点击事件,转到Instrumentation类的startActivity方法。Instrumentation
通过跨进程通信告诉AMS要启动应用的需求。AMS
反馈Launcher,让Launcher进入Paused状态Launcher
进入Paused状态,AMS转到ZygoteProcess类,并通过socket与Zygote通信,告知Zygote需要新建进程。Zygote
fork进程,并调用ActivityThread的main方法,也就是app的入口。ActivityThread
的main方法新建了ActivityThread实例,并新建了Looper实例,开始loop循环。- 同时
ActivityThread
也告知AMS,进程创建完毕,开始创建Application,Provider,并调用Applicaiton的attach,onCreate方法。 - 最后就是创建上下文,通过类加载器加载Activity,调用Activity的
onCreate
方法。
里面涉及到哪些重要的成员呢?都分别负责了什么,比如AMS?
init进程
,Android系统启动后,Zygote并不是第一个进程,而是linux的根进程init进程,然后init进程才会启动Zygote进程。Zygote进程
,所有android进程的父进程,当然也包括SystemServer进程SystemServer进程
,正如名字一样,系统服务进程,负责系统中大大小小的事物,为此也是启动了三员大将(ActivityManagerService,PackageManagerService,WindowManagerService)以及binder线程池。ActivityManagerService
,主要负责系统中四大组件的启动、切换、调度及应用进程的管理和调度等工作,对于一些进程的启动,都会通过Binder通信机制传递给AMS,再处理给Zygote。PackageManagerService
,主要负责应用包的一些操作,比如安装,卸载,解析AndroidManifest.xml,扫描文件信息等等。WindowManagerService
,主要负责窗口相关的一些服务,比如窗口的启动,添加,删除等。Launcher
,桌面应用,也是属于应用,也有自己的Activity,一开机就会默认启动,通过设置Intent.CATEGORY_HOME的Category隐式启动。
启动流程了解后,对我们开发有什么帮助呢?
分析源码的目的一直都不是为了学知识而学,而是理解了这些基础,我们才能更好的解决问题。学习了App的启动流程,我们可以再思考下一些之前没理解透的问题。
1)比如启动优化
,分析启动过程,其实可以优化启动速度的地方有三个地方:
Application的attach方法
,MultiDexApplication会在方法里面会去执行MultiDex逻辑。所以这里可以进行MultiDex优化,比如今日头条方案就是单独启动一个进程的activity去加载MultiDex。Application的onCreate方法
,大量三方库的初始化都在这里进行,所以我们可以开启线程池,懒加载等等。把每个启动任务进行区分,哪些可以子线程运行,哪些有先后顺序。Activity的onCreate方法
,同样进行线程处理,懒加载。或者预创建Activity,提前类加载等等。
2)又比如插件化,通过了解启动流程可以知道哪些地方可以用来hook
,从而完成我们侵入代码,替换Activity的工作。
3)还有上次说过的Activity显示View的过程,我们才知道什么时候进行DecorView
的加载,什么时候进行view绘制等等
RecycleView三问—腾讯真题
Recycleview相比也是每个Android开发者熟得不能再熟的控件了,但是你对他又真的了解多少呢?看看今天的三问你都能答得上来吗?
- 和listview区别
- Recycleview有几级缓存,缓存过程?
- 说说RecyclerView性能优化。
和listview区别
Recycleview布局效果更多
,增加了纵向,表格,瀑布流等效果Recycleview去掉了一些api
,比如setEmptyview,onItemClickListener等等,给到用户更多的自定义可能Recycleview去掉了设置头部底部item的功能
,专向通过viewholder的不同type实现Recycleview实现了一些局部刷新
,比如notifyitemchangedRecycleview自带了一些布局变化的动画效果
,也可以通过自定义ItemAnimator类实现自定义动画效果Recycleview缓存机制更全面
,增加两级缓存,还支持自定义缓存逻辑
Recycleview有几级缓存,缓存过程?
Recycleview有四级缓存,分别是mAttachedScrap(屏幕内),mCacheViews(屏幕外),mViewCacheExtension(自定义缓存),mRecyclerPool(缓存池)
mAttachedScrap(屏幕内)
,用于屏幕内itemview快速重用,不需要重新createView和bindViewmCacheViews(屏幕外)
,保存最近移出屏幕的ViewHolder,包含数据和position信息,复用时必须是相同位置的ViewHolder才能复用,应用场景在那些需要来回滑动的列表中,当往回滑动时,能直接复用ViewHolder数据,不需要重新bindView。mViewCacheExtension(自定义缓存)
,不直接使用,需要用户自定义实现,默认不实现。mRecyclerPool(缓存池)
,当cacheView满了后或者adapter被更换,将cacheView中移出的ViewHolder放到Pool中,放之前会把ViewHolder数据清除掉,所以复用时需要重新bindView。
四级缓存按照顺序需要依次读取。所以完整缓存流程是:
- 保存缓存流程:
- 插入或是删除
itemView
时,先把屏幕内的ViewHolder保存至AttachedScrap
中 - 滑动屏幕的时候,先消失的itemview会保存到
CacheView
,CacheView大小默认是2,超过数量的话按照先入先出原则,移出头部的itemview保存到RecyclerPool缓存池
(如果有自定义缓存就会保存到自定义缓存里),RecyclerPool缓存池会按照itemview的itemtype
进行保存,每个itemTyep缓存个数为5个,超过就会被回收。
- 获取缓存流程:
- AttachedScrap中获取,通过pos匹配holder——>获取失败,从
CacheView
中获取,也是通过pos获取holder缓存 ——>获取失败,从自定义缓存
中获取缓存——>获取失败,从mRecyclerPool
中获取 ——>获取失败,重新创建viewholder
——createViewHolder并bindview。
需要注意的是,如果从缓存池找到缓存,还需要重新bindview。
说说RecyclerView性能优化。
bindViewHolder
方法是在UI线程进行的,此方法不能耗时操作,不然将会影响滑动流畅性。比如进行日期的格式化。- 对于新增或删除的时候,可以使用
diffutil
进行局部刷新,少用全局刷新 - 对于
itemVIew
进行布局优化,比如少嵌套等。 - 25.1.0 (>=21)及以上使用
Prefetch
功能,也就是预取功能,嵌套时且使用的是LinearLayoutManager,子RecyclerView可通过setInitialPrefatchItemCount设置预取个数 - 加大
RecyclerView缓存
,比如cacheview大小默认为2,可以设置大点,用空间来换取时间,提高流畅度 - 如果高度固定,可以设置
setHasFixedSize(true)
来避免requestLayout浪费资源,否则每次更新数据都会重新测量高度。
void onItemsInsertedOrRemoved() {
if (hasFixedSize) layoutChildren();
else requestLayout();
}
- 如果多个
RecycledView
的 Adapter 是一样的,比如嵌套的 RecyclerView 中存在一样的 Adapter,可以通过设置RecyclerView.setRecycledViewPool(pool);
来共用一个RecycledViewPool
。这样就减少了创建VIewholder的开销。 - 在RecyclerView的元素比较高,一屏只能显示一个元素的时候,第一次滑动到第二个元素会卡顿。这种情况就可以通过设置额外的缓存空间,重写
getExtraLayoutSpace
方法即可。
new LinearLayoutManager(this) {
@Override
protected int getExtraLayoutSpace(RecyclerView.State state) {
return size;
}
};
- 设置
RecyclerView.addOnScrollListener();
来在滑动过程中停止加载的操作。 - 减少对象的创建,比如设置监听事件,可以全局创建一个,所有view公用一个listener,并且放到
CreateView
里面去创建监听,因为CreateView调用要少于bindview。这样就减少了对象创建所造成的消耗 - 用
notifyDataSetChange
时,适配器不知道整个数据集中的那些内容以及存在,再重新匹配ViewHolder
时会花生闪烁。设置adapter.setHasStableIds(true),并重写getItemId()
来给每个Item一个唯一的ID,也就是唯一标识,就使itemview的焦点固定,解决了闪烁问题。
面试前做好准备战!
接下来将分享面试的一个复习路线,如果你也在准备面试但是不知道怎么高效复习,可以参考一下我的复习路线,有任何问题也欢迎一起互相交流,加油吧!
这里给大家提供一个方向,进行体系化的学习:
1、看视频进行系统学习
前几年的Crud经历,让我明白自己真的算是菜鸡中的战斗机,也正因为Crud,导致自己技术比较零散,也不够深入不够系统,所以重新进行学习是很有必要的。我差的是系统知识,差的结构框架和思路,所以通过视频来学习,效果更好,也更全面。关于视频学习,个人可以推荐去B站进行学习,B站上有很多学习视频,唯一的缺点就是免费的容易过时。
另外,我自己也珍藏了好几套视频,有需要的我也可以分享给你。
2、进行系统梳理知识,提升储备
客户端开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。
系统学习方向:
- 架构师筑基必备技能:深入Java泛型 注解深入浅出 并发编程 数据传输与序列化 Java虚拟机原理 反射与类加载 动态代理 高效IO
- Android高级UI与FrameWork源码:高级UI晋升 Framework内核解析 Android组件内核 数据持久化
- 360°全方面性能调优:设计思想与代码质量优化 程序性能优化 开发效率优化
- 解读开源框架设计思想:热修复设计 插件化框架解读 组件化框架设计 图片加载框架 网络访问框架设计 RXJava响应式编程框架设计 IOC架构设计 Android架构组件Jetpack
- NDK模块开发:NDK基础知识体系 底层图片处理 音视频开发
- 微信小程序:小程序介绍 UI开发 API操作 微信对接
- Hybrid 开发与Flutter:Html5项目实战 Flutter进阶
知识梳理完之后,就需要进行查漏补缺,所以针对这些知识点,我手头上也准备了不少的电子书和笔记,这些笔记将各个知识点进行了完美的总结。
3、读源码,看实战笔记,学习大神思路
“编程语言是程序员的表达的方式,而架构是程序员对世界的认知”。所以,程序员要想快速认知并学习架构,读源码是必不可少的。阅读源码,是解决问题 理解事物,更重要的:看到源码背后的想法;程序员说:读万行源码,行万种实践。
主要内含微信 MMKV 源码、AsyncTask 源码、Volley 源码、Retrofit源码、OkHttp 源码等等。
4、面试前夕,刷题冲刺
面试的前一周时间内,就可以开始刷题冲刺了。请记住,刷题的时候,技术的优先,算法的看些基本的,比如排序等即可,而智力题,除非是校招,否则一般不怎么会问。
关于面试刷题,我个人也准备了一套系统的面试题,帮助你举一反三。
总结
改变人生,没有什么捷径可言,这条路需要自己亲自去走一走,只有深入思考,不断反思总结,保持学习的热情,一步一步构建自己完整的知识体系,才是最终的制胜之道,也是程序员应该承担的使命。
还有耗时一年多整理的一系列Android学习资源:Android源码解析、Android第三方库源码笔记、Android进阶架构师七大专题学习、历年BAT面试题解析包、Android大佬学习笔记
等等。