生还是死?Android 进程优先级详解

在收到 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 。

空进程

在任何层次中,空进程都是最低优先级的。如果不属于以上类别,那它就是这种。这里没有活跃的组件,只是出于缓存的目的而被保留(为了更加有效地使用内存而不是完全释放掉),只要 Android 需要可以随时杀掉它们。

注意事项

当我们谈论进程优先级的时候是以 activity、service 这样的组件来说的,但请记住这些优先级是在进程的级别上,不是组件级别上。只要一个组件(比如一个前台服务)就会将整个进程变为前台进程。绝大多数应用是单进程的,如果你有生命周期差异很大的不同部分或者某个部分非常重量型,那么强烈建议你把它们分为不同的进程,让重量级进程尽早被回收。

同样重要的是,你的进程属于什么类别是组件层面发生的事情决定的。这意味着把非常重要的长时间运行的操作放在 activity 所在进程的一个独立线程中的做法,在进程突然变成后台进程的时候可能会遇到问题。使用你能用到的工具(一个服务或基于优先级的前台服务)来确保系统知道你在做什么。

与别人友好相处,把用户放在心里

整个系统这样工作都是为了用户。做个好公民,做好你的应用,始终让自己工作在合适的优先级上。请记住,作为一个开发者,你使用的手机可能比你用户的最差手机快得多得多,你可能从来不会看到可见进程被杀死,远少于服务进程,但是这不意味着它不会发生!

我仍然建议你购买一个非常低端的 Android 设备用于测试,同时你也可以在高端设备上测试被杀掉时应用是如何响应的。要在包级别杀掉应用,请使用:

adb shell am force-stop com.example.packagename

如果你有多个进程,可以在第二栏找到进程 id(PID)(如,下面第一个数字):

adb shell ps | grep com.example.packagename

然后这样杀掉:

adb shell kill PID

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

在这里我和身边一些朋友特意整理了一份快速进阶为Android高级工程师的系统且全面的学习资料。涵盖了Android初级——Android高级架构师进阶必备的一些学习技能。

附上:我们之前因为秋招收集的二十套一二线互联网公司Android面试真题(含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

[外链图片转存中…(img-hV3Tc1xt-1713687088901)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值