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

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

英文原址:Who lives and who dies? Process priorities on Android

让我们面对现实:移动设备上没有无限的内存、无限的电池或者其它无限的资源。这对应用而言意味着你应该把进程死亡作为应用生命周期的一个自然过程对待。最重要的是确保杀死进程及内存回收不会对用户造成负面影响。事实上,Android 中的多数进程架构都是为了确保特定的顺序而特别设计的,并按重要性层次遵循一组模式。

Android 进程层次

你会发现最重要的进程被称为前台进程,然后依次是任何可见进程、服务进程、后台进程,最后是空进程。这个文档中有详细描述,这里我们将进一步展开。

注意,当我们谈论特定组件(服务、activity)时,Android 只杀死进程,而不是组件。当然,这不会阻止通常的垃圾回收进程(它要回收没有任何引用的对象的内存),不过这是另一个主题了。

前台进程

你会想正在与用户交互的东西是最重要的需要保证活着的,这应该完全正确。但是“正在与用户交互”这个定义有点模糊。当前的前台 Activity 毫无争议属于这一类,它是已经调用了 onResume() 方法但还没有收到 onPause() 调用的 Activity .

一些 activity 在依靠他们自己的同时,也可能依赖 bound service。任何进程,如果它持有一个绑定到前台 activity 的服务,那么它也被赋予了同样的前台优先级。这完全符合直觉,如果前台 activity 认为和那个服务保持持久连接很重要,那么保持这个服务活着就对 activity 和 Android 很重要。对于正在与前台服务交互的 content provider 也是如此。

但是谁说用户能察觉到的只有 activity ?如果正在播放的音乐突然停止或导航方向突然消失,我一定会很恼火。幸好,Android 可以让服务使用 startForeground() 方法成为高优先级前台服务。这绝对是媒体播放的最佳实践,但是这里要问一个重要问题“如果服务停止了,用户会立刻察觉到吗?”。前台服务应该仅被用于关键的、可被立刻察觉的场景。

注意:要成为前台服务需要在服务中包含一个通知以便让用户注意到这个服务正在运行。如果你觉得你的使用场景不需要这个通知,那么前台服务对你可能不是正确的选择(是的,成为前台服务并不要求一定运行在后台,见下文)。

可见进程

等下,我想我已经谈到了当前的 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 生命周期)。
  1. START_REDELIVER_INTENT 是为那些在被杀死之后重启时重新获得 Intent 的服务的,直到你用传递给 onStartCommand() 方法的 startId 参数调用 stopSelf() 为止。这里你会使用 Intent 和 startId 作为队列完成工作。
  1. START_NOT_STICKY 用于那些杀掉也没关系的服务。这适合那些管理周期性任务的服务,它们只是等待下一个时间窗口工作。

后台进程

比如说你的 Activity 一开始是前台 Activity,但是用户点了 home 键导致 onStop() 方法被调用。假设你之前一直是高优先级进程类别,这时你的进程将变为后台进程类别。在一般操作场景下,设备上的许多内存就是用在这上面的,让你重新回到之前打开过的某个 activity 。

Android 不是为了杀而杀的(记住:从头启动是有代价的),所以这些进程会保留一段时间,直到更高优先级进程需要内存的时候才被回收,并且是按照最近最少使用顺序(最老的会被优先回收)。然而,当他们被杀掉的时候和可见 activity 处理情况一样,你应该能够在不丢失用户状态的情况下重建这些 activity 。

空进程

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

注意事项

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

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

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,376评论 6 491
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,126评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,966评论 0 347
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,432评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,519评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,792评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,933评论 3 406
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,701评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,143评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,488评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,626评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,292评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,896评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,742评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,977评论 1 265
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,324评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,494评论 2 348

推荐阅读更多精彩内容