生还是死?Android 进程优先级详解,Android多态实现原理
[]( )前言让我们面对现实:移动设备上没有无限的内存、无限的电池或者其它无限的资源。这对应用而言意味着你应该把进程死亡作为应用生命周期的一个自然过程对待。最重要的是确保杀死进程及内存回收不会对用户造成负面影响。事实上,Android 中的多数进程架构都是为了确保特定的顺序而特别设计的,并按重要性层次遵循一组模式。
[]( )Android 进程层次
你会发现最重要的进程被称为前台进程,然后依次是任何可见进程、服务进程、后台进程,最后是空进程。这个文档中有详细描述,这里我们将进一步展开。
注意,当我们谈论特定组件(服务、activity)时,Android 只杀死进程,而不是组件。当然,这不会阻止通常的垃圾回收进程(它要回收没有任何引用的对象的内存),不过这是另一个主题了。
[]( )前台进程
你会想正在与用户交互的东西是最重要的需要保证活着的,这应该完全正确。但是“正在与用户交互”这个定义有点模糊。当前的前台 Activity?毫无争议属于这一类,它是已经调用了 onResume()?方法但还没有收到 onPause()?调用的 Activity 。
一些 activity 在依靠他们自己的同时,也可能依赖 bound service 。**任何进程,如果它持有一个绑定到前台 activity 的服务,那么它也被赋予了同样的前台优先级。**这完全符合直觉,如果前台 activity 认为和那个服务保持持久连接很重要,那么保持这个服务活着就对 activity 和 Android 很重要。对于正在与前台服务交互的 content provider 也是如此。
但是谁说用户能察觉到的只有 activity ?如果正在播放的音乐突然停止或导航方向突然消失,我一定会很恼火。幸好,Android 可以让服务使用 startForeground()?方法成为高优先级前台服务。这绝对是媒体播放的最佳实践,但是这里要问一个重要问题“如果服务停止了,用户会立刻察觉到吗?”。前台服务应该仅被用于关键的、可被立刻察觉的场景。
**注意:**要成为前台服务需要在服务中包含一个通知以便让用户注意到这个服务正在运行。如果你觉得你的使用场景不需要这个通知,那么前台服务对你可能不是正确的选择(是的,成为前台服务并不要求一定运行在后台,见下文)。
在接收关键生命周期方法时会让一个进程临时提升为前台进程,包括任何服务的生命周期方法(onCreate(), onStartCommand()?和 onDestroy()) 和任何广播接收器 onReceive()?方法。这样做确保了这些组件的操作是有效的原子操作,每个组件都能执行完成而不被杀掉。
[]( )可见进程
等下,我想我已经谈到了当前的 activity?你会发现?activity 可见的时候不一定在前台。一个简单的例子是前台的 activity 使用对话框启动了一个新的 activity 或者一个透明 activity 。另一个例子是当你调用运行时权限对话框时(事实上它就是一个 activity!)。
在收到 onStart()?和收到 onStop()?方法期间的 activity 是可见 activity 。在这两个方法调用之间,你可以做所有可见 activity 能做的事情(实时更新屏幕等)。
和前台 activity 类似,可见 activity 的 bound service 和 content provider 也处于可见进程状态。这同样是为了保证使用中的 activity 所依赖的进程不会被过早地杀掉。
但请记住,只是可见并不意味着不能被杀掉。如果来自前台进程的内存压力过大,可见进程仍有可能被杀掉。从用户的角度看,这意味着当前 activity 背后的可见 activity 会被黑屏代替。当然,如果你正确地重建你的 activity ,在前台 activity 关闭之后你的进程和 activity 会立刻恢复而没有数据损失。
**注意:**你的 activity 和进程即使可见也可能被杀掉是因为startActivityForResult()+onActivityResult()?或 requestPermissions()+onRequestPermissionsResult()?流程没有获得回调类的实例。如果你的整个进程死了,那么所有的回调类实例也死了。如果你看到使用回调方式的库,你应该意识到这在低内存压力情况下无法完成。
[]( )服务进程
如果你的进程不属于以上两种类别,而你有一个启动的服务(started service),那么它被看作是一个服务进程。对于许多在后台做处理(如加载数据)而没有立即成为前台服务的应用都属于这种情况。
这没有问题!绝大多数情况,这是后台处理的最佳方式。这种进程只有在前面讲的可见进程和前台进程做了太多事情需要更多资源的时候才会被杀掉。
请特别注意从 onStartCommand()?返回的常量,如果你的服务由于内存压力被杀掉,它表示控制什么发生什么:
1、**START_STICKY?**表示你希望系统可用的时候自动重启你的服务,但你不关心是否能获得最后一次的 Intent (例如,你可以重建自己的状态或者控制自己的 start/stop 生命周期)。
2、START_REDELIVER_INTENT?是为那些在被杀死之后重启时重新获得 Intent 的服务的,直到你用传递给 onStartCommand() 方法的 startId 参数调用 stopSelf()?为止。这里你会使用 Intent 和 startId 作为队列完成工作。
3、**START_NOT_STICKY?**用于那些杀掉也没关系的服务。这适合那些管理周期性任务的服务,它们只是等待下一个时间窗口工作。
[]( )后台进程
比如说你的 Activity 一开始是前台 Activity,但是用户点了 home 键导致 onStop()?方法被调用。假设你之前一直是高优先级进程类别,这时你的进程将变为后台进程类别。在一般操作场景下,设备上的许多内存就是用在这上面的,让你重新回到之前打开过的某个 activity 。
Android 不是为了杀而杀的(记住:从头启动是有代价的),所以这些进程会保留一段时间,直到更高优先级进程需要内存的时候才被回收,并且是按照最近最少使用顺序(最老的会被优先回收)。然而,当他们被杀掉的时候和可见 activity 处理情况一样,你应该能够在不丢失用户状态的情况下重建这些 activity 。
题外话
我在一线互联网企业工作十余年里,指导过不少同行后辈。帮助很多人得到了学习和成长。
我意识到有很多经验和知识值得分享给大家,也可以通过我们的能力和经验解答大家在IT学习中的很多困惑,所以在工作繁忙的情况下还是坚持各种整理和分享。但苦于知识传播途径有限,很多程序员朋友无法获得正确的资料得到学习提升,故此将并将重要的Android进阶资料包括自定义view、性能优化、MVC与MVP与MVVM三大框架的区别、NDK技术、阿里面试题精编汇总、常见源码分析等学习资料免费分享出来。
【Android学习PDF+学习视频+面试文档+知识点笔记】
【Android思维脑图(技能树)】
知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。
需要的朋友,可以点赞关注+转发”前往免费领取!
希望我能够用我的力量帮助更多迷茫、困惑的朋友们,帮助大家在IT道路上学习和发展~
https://blog.51cto.com/u_15465267/4842231
页:
[1]