Android进程生命周期与ADJ

一、 进程生命周期

Android系统将尽量长时间地保持应用进程,但为了新建进程或运行更重要的进程,最终需要清除旧进程来回收内存。 为了确定保留或终止哪些进程,系统会根据进程中正在运行的组件以及这些组件的状态,将每个进程放入“重要性层次结构”中。 必要时,系统会首先消除重要性最低的进程,然后是清除重要性稍低一级的进程,依此类推,以回收系统资源。

进程的重要性,划分5级:

前台进程(Foreground process)
可见进程(Visible process)
服务进程(Service process)
后台进程(Background process)
空进程(Empty process)
前台进程的重要性最高,依次递减,空进程的重要性最低,下面分别来阐述每种级别的进程

1.1 Foreground process

用户当前操作所必需的进程。通常在任意给定时间前台进程都为数不多。只有在内存不足以支持它们同时继续运行这一万不得已的情况下,系统才会终止它们。

  • 拥有用户正在交互的 Activity(已调用onResume())
  • 拥有某个 Service,后者绑定到用户正在交互的 Activity
  • 拥有正在“前台”运行的 Service(服务已调用 startForeground())
  • 拥有正执行一个生命周期回调的 Service(onCreate()、onStart() 或 onDestroy())
  • 拥有正执行其 onReceive() 方法的 BroadcastReceiver

1.2 Visible process

没有任何前台组件、但仍会影响用户在屏幕上所见内容的进程。可见进程被视为是极其重要的进程,除非为了维持所有前台进程同时运行而必须终止,否则系统不会终止这些进程。

  • 拥有不在前台、但仍对用户可见的 Activity(已调用onPause())。
  • 拥有绑定到可见(或前台)Activity 的 Service

1.3 Service process

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

  • 正在运行startService()方法启动的服务,且不属于上述两个更高类别进程的进程。

1.4 Background process

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

  • 对用户不可见的Activity的进程(已调用Activity的onStop()方法)

1.5 Empty process

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

  • 不含任何活动应用组件的进程

二、Lowmemorykiller

2.1 ADJ级别

定义在ProcessList.java文件,oom_adj划分为16级,从-1000到1001之间取值。

ADJ级别 取值 解释
UNKNOWN_ADJ 1001 一般指将要会缓存进程,无法获取确定值
CACHED_APP_MAX_ADJ 999 不可见进程的adj最大值
CACHED_APP_MIN_ADJ 900 不可见进程的adj最小值
SERVICE_B_AD 800 B List中的Service(较老的、使用可能性更小)
PREVIOUS_APP_ADJ 700 上一个App的进程(往往通过按返回键)
HOME_APP_ADJ 600 Home进程
SERVICE_ADJ 500 服务进程(Service process)
HEAVY_WEIGHT_APP_ADJ 400 后台的重量级进程,system/rootdir/init.rc文件中设置
BACKUP_APP_ADJ 300 备份进程
PERCEPTIBLE_APP_ADJ 200 可感知进程,比如后台音乐播放
VISIBLE_APP_ADJ 100 可见进程(Visible process)
FOREGROUND_APP_ADJ 50 前台进程(Foreground process)
PERSISTENT_SERVICE_ADJ -700 关联着系统或persistent进程
PERSISTENT_PROC_ADJ -800 系统persistent进程,比如telephony
SYSTEM_ADJ -900 系统进程
NATIVE_ADJ -1000 native进程(不被系统管理)

这里按照Android对进程的分类,粗略划分一下不同oom_adj对应的场景

image.png
14.png

2.2 进程state级别

定义在ActivityManager.java文件,process_state,从-1到20之间取值。

state级别 取值 解释
PROCESS_STATE_NONEXISTENT 20 进程不存在
PROCESS_STATE_CACHED_EMPTY 19 进程处于cached状态,且为空进程
PROCESS_STATE_CACHED_RECENT 18 与recent中关联的缓存进程
PROCESS_STATE_CACHED_ACTIVITY_CLIENT 17 托管存在客户端的Activity的进程
PROCESS_STATE_CACHED_ACTIVITY 16 后台托管有Activity的进程
PROCESS_STATE_LAST_ACTIVITY 15 后台进程,但是托管了用户最后运行的Activity
PROCESS_STATE_HOME 14 Home进程
PROCESS_STATE_HEAVY_WEIGHT 13 重量级进程
PROCESS_STATE_TOP_SLEEPING 12 和PROCESS_STATE_TOP一样,但是区别是手机处于sleeping状态
PROCESS_STATE_RECEIVER 11 后台进程,且正在运行receiver
PROCESS_STATE_SERVICE 10 后台进程,且正在运行service
PROCESS_STATE_BACKUP 9 备份进程
PROCESS_STATE_TRANSTENT_BACKGROUND 8 对于用户比较重要的后台进程
PROCESS_STATE_IMPORTANT_BACKGROUND 7 对用户很重要的进程,用户可感知其存在
PROCESS_STATE_IMPORTANT_FOREGROUND 6 对于用户比较重要的前台进程
PROCESS_STATE_BOUND_FOREGROUND_SERVICE 5 通过bind的形式托管前台服务的进程
PROCESS_STATE_FOREGROUND_SERVICE 4 托管前台服务的进程
PROCESS_STATE_BOUND_TOP 3 进程绑定到一个TOP应用。它的排名低于SERVICE_LOCATION
PROCESS_STATE_TOP 2 涵盖了用户可见活动的进程
PROCESS_STATE_PERSISTENT_UI 1 系统persistent进程,正在进行UI相关的操作
PROCESS_STATE_PERSISTENT 0 系统persistent进程
PROCESS_STATE_UNKNOWN -1 不存在的进程

2.3 lmk策略

Lowmemorykiller根据当前可用内存情况来进行进程释放,总设计了6个级别,即Lowmemorykiller的杀进程的6档,如下:

  1. CACHED_APP_MAX_ADJ
  2. CACHED_APP_MIN_ADJ
  3. BACKUP_APP_ADJ
  4. PERCEPTIBLE_APP_ADJ
  5. VISIBLE_APP_ADJ
  6. FOREGROUND_APP_ADJ

系统内存从很宽裕到不足,Lowmemorykiller也会相应地从CACHED_APP_MAX_ADJ(第1档)开始杀进程,如果内存还不足,那么会杀CACHED_APP_MIN_ADJ(第2档),不断深入,直到满足内存阈值条件。

三、 进程调度adj算法

ADJ算法的核心方法:

  • updateOomAdjLocked:更新adj,当目标进程为空,或者被杀则返回false;否则返回true;
  • computeOomAdjLocked:计算adj,返回计算后RawAdj值;
  • applyOomAdjLocked:应用adj,当需要杀掉目标进程则返回false;否则返回true。

调整adj的3的方法,最为常见的方法便是updateOomAdjLocked,这也是其他各个方法在需要更新adj时会调用的方法。

四、ADJ的更新时机

哪些场景下都会触发updateOomAdjLocked来更新进程adj:

4.1 Activity

  • ASS.realStartActivityLocked: 启动Activity
  • AS.resumeTopActivityInnerLocked: 恢复栈顶Activity
  • AS.finishCurrentActivityLocked: 结束当前Activity
  • AS.destroyActivityLocked: 摧毁当前Activity

4.2 Service

位于ActiveServices.java

  • realStartServiceLocked: 启动服务
  • bindServiceLocked: 绑定服务(只更新当前app)
  • unbindServiceLocked: 解绑服务 (只更新当前app)
  • bringDownServiceLocked: 结束服务 (只更新当前app)
  • sendServiceArgsLocked: 在bringup或则cleanup服务过程调用 (只更新当前app)

4.3 broadcast

  • BQ.processNextBroadcast: 处理下一个广播
  • BQ.processCurBroadcastLocked: 处理当前广播
  • BQ.deliverToRegisteredReceiverLocked: 分发已注册的广播 (只更新当前app)

4.4 ContentProvider

  • AMS.removeContentProvider: 移除provider
  • AMS.publishContentProviders: 发布provider (只更新当前app)
  • AMS.getContentProviderImpl: 获取provider (只更新当前app)

4.5 Process

位于ActivityManagerService.java

  • setSystemProcess: 创建并设置系统进程
  • addAppLocked: 创建persistent进程
  • attachApplicationLocked: 进程创建后attach到system_server的过程;
  • trimApplications: 清除没有使用app
  • appDiedLocked: 进程死亡
  • killAllBackgroundProcesses: 杀死所有后台进程.即(ADJ>9或removed=true的普通进程)
  • killPackageProcessesLocked: 以包名的形式 杀掉相关进程;

五、updateOomAdjLocked

5.1 重要方法

/frameworks/base/services/core/java/com/android/server/am/ActivityManagerService.java
      @GuardedBy("this")
      final void updateOomAdjLocked(String oomAdjReason) {
          mOomAdjuster.updateOomAdjLocked(oomAdjReason);
      }

      /*
       * Update OomAdj for a specific process and its reachable processes.
       * @param app The process to update
       * @param oomAdjReason
       */
      @GuardedBy("this")
      final void updateOomAdjLocked(ProcessRecord app, String oomAdjReason) {
          mOomAdjuster.updateOomAdjLocked(app, oomAdjReason);
      }

/frameworks/base/services/core/java/com/android/server/am/OomAdjuster.java
      

       /**
       * Update OomAdj for all processes in LRU list
       */
      @GuardedBy("mService")
      void updateOomAdjLocked(String oomAdjReason) {
//获取栈顶Activity
          final ProcessRecord topApp = mService.getTopAppLocked();
          updateOomAdjLockedInner(oomAdjReason, topApp , null, null, true, true);
      }
      /**
       * Update OomAdj for all processes within the given list (could be partial), or the whole LRU
       * list if the given list is null; when it's partial update, each process's client proc won't
       * get evaluated recursively here.
       */
      @GuardedBy("mService")
      private void updateOomAdjLockedInner(String oomAdjReason, final ProcessRecord topApp,
              ArrayList<ProcessRecord> processes, ActiveUids uids, boolean potentialCycles,
              boolean startProfiling) {
        private final boolean computeOomAdjLocked(ProcessRecord app, int cachedAdj,
              ProcessRecord topApp, boolean doingAll, long now, boolean cycleReEval,
              boolean computeClients) {
       /** Applies the computed oomadj, procstate and sched group values and freezes them in set* */
      @GuardedBy("mService")
      private final boolean applyOomAdjLocked(ProcessRecord app, boolean doingAll, long now,
              long nowElapsed) {

等重要方法。

  • updateOomAdjLocked在应用进程的组件运行状态发生改变时被调用,比如有Service启动,有广播接收者收到广播,有Activity启动等,这很好理解,因为进程重要性的计算就依赖于组件运行状态,既然组件运行状态发生了改变,就应该实时更新;
  • computeOomAdjLocked根据一定规则计算出三个状态值,这个规则跟Android将进程划分的5个优先级有关系,即前台进程、可见进程、服务进程、后台进程、空进程,这里不详细说明;
  • applyOomAdjLocked将computeOomAdjLocked计算出的三个状态值应用起来,即真正发挥这三个状态值的作用。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容