Android多进程介绍

在Android中,默认情况下,同一应用的所有组件均运行在同一进程中,且大多数应用都不会改变这一点。不过,单进程开发并不是Android应用的全部,今天我们就来说说Android中的多进程开发以及多进程的使用场景。

学习Android的同学注意了!!!

学习过程中遇到什么问题或者想获取学习资源的话,欢迎加入Android学习交流群,群号码:364595326  我们一起学Android!

多进程介绍

本篇文章内容基于Android Developer官网

进程

我们都知道Android系统是基于Linux改造而来的,进程系统也是一脉相承,进程,其实就是程序的具体实现。当程序第一次启动,Android会启动一个Linux进程(具体由Zygote fork出来)以及一个主线程,默认的情况下,所有组件都将运行在该进程内。同一个应用由系统分配一个独立的Linux账户,该应用的产生的所有进程,都会是这同一个Linux账户。

使用多进程

在开发中,我们通常会使用修改清单文件的android:process来达到多进程的目的。activity、service、receiver和provider均支持android:process属性,此属性可以指定该组件应在哪个进程运行。如果android:process的value值以冒号开头的话,那么该进程就是私有进程,如果是以其他字符开头,那么就是公有进程,拥有相同ShareUID的不同应用可以跑在同一进程里,后续我会专门针对公私有进程做个试验。另外,我们还可以通过设置application的android:process属性,来设置所有组件的默认进程。

至于创建进程的具体源码分析,网上有一篇很详细的文章,在这就不重复造轮子了,有需要的朋友可以前往理解Android进程创建流程

还有一种方法开启进程,是通过JNI,利用C/C++,调用fork()方法来生成子进程,一般开发者会利用这种方法来做一些daemon进程,来实现防杀,保活等效果,不过不是太推荐,这么做,毕竟Android生态系统需要大家维护。

进程生命周期

刚刚聊了一下进程的“生”,作为一个生命周期,是时候该聊聊进程的“死”了。这里再次呼吁一下大家能正视进程的“死”,合理的利用多进程,适当的杀死不必要的进程才是本篇文章所关注的焦点,我们不要把“永生”作为自己的实现目标,Android设备内存就那么大,就像地球一样,大家都永生了,生态系统就会破坏。那么Android系统是如何维护这个生态系统的呢?

其实也是类似于现实生活中的优胜略汰,Android利用重要性层次结构,就是将最重要的保留,杀掉不重要的进程。Android将重要性层次结构分为5个层级,分为了:(以下5级描述节选自Android进程生命周期

前台进程

用户当前操作所必需的进程。如果一个进程满足以下任一条件,即视为前台进程:

托管用户正在交互的Activity(已调用Activity的onResume()方法)

托管某个Service,后者绑定到用户正在交互的Activity

托管正在“前台”运行的Service(服务已调用startForeground())

托管正执行一个生命周期回调的Service(onCreate()、onStart()或onDestroy())

托管正执行其onReceive()方法的BroadcastReceiver

通常,在任意给定时间前台进程都为数不多。只有在内存不足以支持它们同时继续运行这一万不得已的情况下,系统才会终止它们。 此时,设备往往已达到内存分页状态,因此需要终止一些前台进程来确保用户界面正常响应。

这就需要依靠系统的资源。

可见进程

没有任何前台组件、但仍会影响用户在屏幕上所见内容的进程。 如果一个进程满足以下任一条件,即视为可见进程:

托管不在前台、但仍对用户可见的Activity(已调用其onPause()方法)。例如,如果前台Activity启动了一个对话框,允许在其后显示上一Activity,则有可能会发生这种情况。

托管绑定到可见(或前台)Activity的Service。

可见进程被视为是极其重要的进程,除非为了维持所有前台进程同时运行而必须终止,否则系统不会终止这些进程。

服务进程

正在运行已使用startService()方法启动的服务且不属于上述两个更高类别进程的进程。尽管服务进程与用户所见内容没有直接关联,但是它们通常在执行一些用户关心的操作(例如,在后台播放音乐或从网络下载数据)。因此,除非内存不足以维持所有前台进程和可见进程同时运行,否则系统会让服务进程保持运行状态。

后台进程

包含目前对用户不可见的Activity的进程(已调用Activity的onStop()方法)。这些进程对用户体验没有直接影响,系统可能随时终止它们,以回收内存供前台进程、可见进程或服务进程使用。 通常会有很多后台进程在运行,因此它们会保存在 LRU (最近最少使用)列表中,以确保包含用户最近查看的Activity的进程最后一个被终止。如果某个Activity正确实现了生命周期方法,并保存了其当前状态,则终止其进程不会对用户体验产生明显影响,因为当用户导航回该Activity时,Activity会恢复其所有可见状态。 有关保存和恢复状态的信息,请参阅Activity文档。

空进程

不含任何活动应用组件的进程。保留这种进程的的唯一目的是用作缓存,以缩短下次在其中运行组件所需的启动时间。 为使总体系统资源在进程缓存和底层内核缓存之间保持平衡,系统往往会终止这些进程。

根据进程中当前活动组件的重要程度,Android 会将进程评定为它可能达到的最高级别。例如,如果某进程托管着服务和可见Activity,则会将此进程评定为可见进程,而不是服务进程。

此外,一个进程的级别可能会因其他进程对它的依赖而有所提高,即服务于另一进程的进程其级别永远不会低于其所服务的进程。 例如,如果进程 A 中的内容提供程序为进程 B 中的客户端提供服务,或者如果进程 A 中的服务绑定到进程 B 中的组件,则进程 A 始终被视为至少与进程 B 同样重要。

由于运行服务的进程其级别高于托管后台Activity的进程,因此启动长时间运行操作的Activity最好为该操作启动服务,而不是简单地创建工作线程,当操作有可能比Activity更加持久时尤要如此。例如,正在将图片上传到网站的Activity应该启动服务来执行上传,这样一来,即使用户退出Activity,仍可在后台继续执行上传操作。使用服务可以保证,无论Activity发生什么情况,该操作至少具备“服务进程”优先级。 同理,广播接收器也应使用服务,而不是简单地将耗时冗长的操作放入线程中。

Low Memory Killer

进程按照状态分完重要性之后,就要开始杀进程了。Android的Low Memory Killer基于Linux的OOM机制,在Linux中,内存是以页面(page)为单位,当申请页面分配不足的时候,系统会通过Low Memory Killer来杀掉bad进程,释放内存。Low Memory Killer会根据进程的adj级别以及所占的内存,来决定是否杀掉该进程,adj越大,占用内存越多,进程越容易被杀掉。

关于adj的分级,我们可以参考ProcessList.java,这里面的常量定义了ADJ的分级。(7.0以后的adj分级与之前的不太一样(Processlist.java-Nougat),这个我们后续可以研究一下具体的改动是什么)

adj分级:

UNKNOWN_ADJ = 16

级别最低级的进程,通常是被缓存的进程,但是系统也不清楚缓存的内容。

CACHED_APP_MAX_ADJ = 15

这是一个只托管不可见的活动的进程,因此可以在没有任何中断的情况下被杀死。

CACHED_APP_MIN_ADJ = 9

缓存进程,没有英文解释。

SERVICE_B_ADJ = 8

不活跃的服务,不想adj=5的服务那么活跃。

PS:这里说一句,在root以后,有的系统优化大师,会把所有服务统一调成adj=8这个级别,来达到内存优化的目的,后面我们会说到。

PREVIOUS_APP_ADJ = 7

被切换的进程,一般是用户前一个使用的进程。两个应用来回切换,那么前一个应用一般adj设置为7。

HOME_APP_ADJ = 6

与主应用程序有交互的进程。

SERVICE_ADJ = 5

活跃的服务进程。

HEAVY_WEIGHT_APP_ADJ = 4

高权重进程

BACKUP_APP_ADJ = 3

正在备份的进程

PERCEPTIBLE_APP_ADJ = 2

可感知进程(通常是前台Service进程)

VISIBLE_APP_ADJ = 1

可见进程

FOREGROUND_APP_ADJ = 0

前台进程

剩下的就是adj值为负数的进程,基本上都是系统集成,不在本文的讨论范围内。负数进程是不会被lmk杀掉的。

如何查看进程优先级

首先通过 adb shell ps 指令查找对应进程的pid

然后通过 adb shell cat /proc/${pid}/oom_adj(设备需要root)返回对应进程的adj值。

还可以把oom_adj替换成oom_score或者oom_score_adj来查看这两项的数值,当oom_adj相同时,LowMemoryKiller会根据oom_score_adj和RSS内存大小来杀掉对应的进程。

查看设备的内存临界值

我们可以通过adb shell cat 查看下面两个文件

/sys/module/lowmemorykiller/parameters/adj

/sys/module/lowmemorykiller/parameters/minfree

(这里请注意,这两个文件是只可以写入的,cat之前请先用chmod赋予权限。)

adj 代表的是oom_score_adj的值,对应的minfree则代表内存临界值。

比如我的测试机小米4C测试机对应的值就是:

adj: 0,58,117,176,529,1000

这个值其实是oom_score_adj的值,用这个值*17再除1000四舍五入取整数,就是对应的adj的值,例如第二个值58即为 58*17/1000 = 1,对应的adj也就是1,所以这6个值对应的adj是0,1,2,3,9,15。1000默认就是15

minfree: 18432,23040,27648,32256,56250,81250

这个值是页值,一页等于4KB,换算成MB大概是72,90,108,126,220,318

当可用内存小于318MB的时候,系统开始杀adj=15的进程,以此类推。

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,088评论 25 707
  • 本篇文章是后台杀死系列的最后一篇,主要探讨一下进程的保活,Android本身设计的时候是非常善良的,它希望进程在不...
    看书的小蜗牛阅读 11,665评论 10 66
  • 什么情况需要使用多线程 常驻后台任务应用 类似音乐类、跑步健身类、手机管家类等长时间需要在后台运行的应用。这些应用...
    TensorFlow开发者阅读 1,807评论 0 3
  • C讀經: - 詩篇31:4 求你救我脫離人為我暗設的網羅.因為你是我的保障。 現在的人越來越聰明,也更加狡猾。在社...
    NCNeverland阅读 141评论 0 0
  • 我不知道挺过了这段苦涩的恋期,我们最终会变成什么样子,至少现在我脑袋里面对未来的幻想逐渐变得苍白无力,我们还可以继...
    小树97阅读 150评论 0 0