應用程式電源管理

在 Android 9 以上版本中,平台可監控應用程式的行為,以防其對裝置的電池續航力造成負面影響。平台會使用及評估設定規則,提供使用者可選擇限制違反規則的應用程式。

在 Android 8.0 以下版本中,Doze、應用程式待命、背景限制和背景位置資訊限制等功能都設有限制。不過,部分應用程式仍會出現不良行為,其中部分行為已在 Android Vitals 中說明。Android 9 推出了作業系統基礎架構,可根據可隨時間更新的設定規則偵測及限制應用程式。

背景限制

使用者可以限制應用程式,或是系統可能會建議應用程式,因為系統偵測到這些應用程式會對裝置健康造成負面影響。

受限制的應用程式:

  • 使用者仍可啟動應用程式。
  • 無法在背景執行工作/鬧鐘或使用網路。
  • 無法執行前景服務。
  • 使用者可以將應用程式變更為未限制的應用程式。

裝置導入者可以為應用程式新增額外限制,以便:

  • 限制應用程式自動重新啟動。
  • 限制服務綁定 (風險極高)。

在背景執行的受限應用程式不應占用任何裝置資源,例如記憶體、CPU 和電池。在使用者未主動使用背景限制應用程式時,這些應用程式不應影響裝置健康狀況。不過,使用者啟動應用程式時,同樣的應用程式應可正常運作。

使用自訂實作

裝置導入者可以繼續使用自訂方法,對應用程式套用限制。

整合應用程式限制

以下各節將概略說明如何在裝置上定義及整合應用程式限制。如果您使用 Android 8.x 以下版本的應用程式限制方法,請仔細查看下列各節,瞭解 Android 9 以上版本的異動。

設定 AppOpsManager 標記

應用程式遭到限制時,請在 AppOpsManager 中設定適當的標記。packages/apps/Settings/src/com/android/settings/fuelgauge/BatteryUtils.java 的程式碼片段範例:

   public void setForceAppStandby(int uid, String packageName,
            int mode) {
        final boolean isPreOApp = isPreOApp(packageName);
        if (isPreOApp) {
       // Control whether app could run in the background if it is pre O app
            mAppOpsManager.setMode(AppOpsManager.OP_RUN_IN_BACKGROUND, uid, packageName, mode);
        }
       // Control whether app could run jobs in the background
        mAppOpsManager.setMode(AppOpsManager.OP_RUN_ANY_IN_BACKGROUND, uid, packageName, mode);
    }

確認 isBackgroundRestricted 傳回值為 true

應用程式受限時,請確認 ActivityManager.isBackgroundRestricted() 會傳回 true

記錄限制原因

應用程式受到限制時,請記錄限制的原因。從 packages/apps/Settings/src/com/android/settings/fuelgauge/batterytip/actions/RestrictAppAction.java 記錄的程式碼片段範例:

mBatteryUtils.setForceAppStandby(mBatteryUtils.getPackageUid(packageName), packageName,AppOpsManager.MODE_IGNORED);
if (CollectionUtils.isEmpty(appInfo.anomalyTypes)) {
  // Only log context if there is no anomaly type
  mMetricsFeatureProvider.action(mContext,
    MetricsProto.MetricsEvent.ACTION_TIP_RESTRICT_APP, packageName,
    Pair.create(MetricsProto.MetricsEvent.FIELD_CONTEXT,metricsKey));
            } else {
  // Log ALL the anomaly types
  for (int type : appInfo.anomalyTypes) {
    mMetricsFeatureProvider.action(mContext,
      MetricsProto.MetricsEvent.ACTION_TIP_RESTRICT_APP, packageName,
      Pair.create(MetricsProto.MetricsEvent.FIELD_CONTEXT, metricsKey),
      Pair.create(MetricsProto.MetricsEvent.FIELD_ANOMALY_TYPE, type));
  }

type 替換為 AnomalyType 中的值。

裝置實作人員可以使用 src/com/android/settings/fuelgauge/batterytip/StatsManagerConfig.java 中定義的常數:

public @interface AnomalyType {
        // This represents an error condition in the anomaly detection.
        int NULL = -1;
         // The anomaly type does not match any other defined type.
        int UNKNOWN_REASON = 0;
         // The application held a partial (screen off) wake lock for a period of time that
         // exceeded the threshold with the screen off when not charging.
        int EXCESSIVE_WAKELOCK_ALL_SCREEN_OFF = 1;
         // The application exceeded the maximum number of wakeups while in the background
         // when not charging.
        int EXCESSIVE_WAKEUPS_IN_BACKGROUND = 2;
         // The application did unoptimized Bluetooth scans too frequently when not charging.
        int EXCESSIVE_UNOPTIMIZED_BLE_SCAN = 3;
         // The application ran in the background for a period of time that exceeded the
         // threshold.
        int EXCESSIVE_BACKGROUND_SERVICE = 4;
         // The application exceeded the maximum number of wifi scans when not charging.
        int EXCESSIVE_WIFI_SCAN = 5;
         // The application exceed the maximum number of flash writes
        int EXCESSIVE_FLASH_WRITES = 6;
         // The application used more than the maximum memory, while not spending any time
         // in the foreground.
        int EXCESSIVE_MEMORY_IN_BACKGROUND = 7;
         // The application exceeded the maximum percentage of frames with a render rate of
         // greater than 700ms.
        int EXCESSIVE_DAVEY_RATE = 8;
         // The application exceeded the maximum percentage of frames with a render rate
         // greater than 16ms.
        int EXCESSIVE_JANKY_FRAMES = 9;
         // The application exceeded the maximum cold start time - the app has not been
         // launched since last system start, died or was killed.
        int SLOW_COLD_START_TIME = 10;
         // The application exceeded the maximum hot start time - the app and activity are
         // already in memory.
        int SLOW_HOT_START_TIME = 11;
         // The application exceeded the maximum warm start time - the app was already in
         // memory but the activity wasn't created yet or was removed from memory.
        int SLOW_WARM_START_TIME = 12;
         // The application exceeded the maximum number of syncs while in the background.
        int EXCESSIVE_BACKGROUND_SYNCS = 13;
         // The application exceeded the maximum number of gps scans while in the background.
        int EXCESSIVE_GPS_SCANS_IN_BACKGROUND = 14;
         // The application scheduled more than the maximum number of jobs while not charging.
        int EXCESSIVE_JOB_SCHEDULING = 15;
         // The application exceeded the maximum amount of mobile network traffic while in
         // the background.
        int EXCESSIVE_MOBILE_NETWORK_IN_BACKGROUND = 16;
         // The application held the WiFi lock for more than the maximum amount of time while
         // not charging.
        int EXCESSIVE_WIFI_LOCK_TIME = 17;
         // The application scheduled a job that ran longer than the maximum amount of time.
        int JOB_TIMED_OUT = 18;
         // The application did an unoptimized Bluetooth scan that exceeded the maximum
         // time while in the background.
        int LONG_UNOPTIMIZED_BLE_SCAN = 19;
         // The application exceeded the maximum ANR rate while in the background.
        int BACKGROUND_ANR = 20;
         // The application exceeded the maximum crash rate while in the background.
        int BACKGROUND_CRASH_RATE = 21;
         // The application exceeded the maximum ANR-looping rate.
        int EXCESSIVE_ANR_LOOPING = 22;
         // The application exceeded the maximum ANR rate.
        int EXCESSIVE_ANRS = 23;
         // The application exceeded the maximum crash rate.
        int EXCESSIVE_CRASH_RATE = 24;
         // The application exceeded the maximum crash-looping rate.
        int EXCESSIVE_CRASH_LOOPING = 25;
         // The application crashed because no more file descriptors were available.
        int NUMBER_OF_OPEN_FILES = 26;
    }

當使用者或系統移除應用程式限制時,您必須記錄移除限制的原因。從 packages/apps/Settings/src/com/android/settings/fuelgauge/batterytip/actions/UnrestrictAppAction.java 記錄的程式碼片段範例:

public void handlePositiveAction(int metricsKey) {
        final AppInfo appInfo = mUnRestrictAppTip.getUnrestrictAppInfo();
        // Clear force app standby, then app can run in the background
        mBatteryUtils.setForceAppStandby(appInfo.uid, appInfo.packageName,
                AppOpsManager.MODE_ALLOWED);
        mMetricsFeatureProvider.action(mContext,
                MetricsProto.MetricsEvent.ACTION_TIP_UNRESTRICT_APP, appInfo.packageName,
                Pair.create(MetricsProto.MetricsEvent.FIELD_CONTEXT, metricsKey));
    }

測試版應用程式限制

如要測試 Android 9 以上版本的應用程式限制行為,請使用下列任一指令:

  • 將應用程式設為受限制:
    appops set package-name RUN_ANY_IN_BACKGROUND ignore
  • 解除應用程式的限制並還原預設行為:
    appops set package-name RUN_ANY_IN_BACKGROUND allow
  • 讓背景中的應用程式立即進入休眠狀態:
    am make-uid-idle [--user user-id | all | current] package-name
  • 將套件新增至 tempwhitelist 一段時間:
    cmd deviceidle tempwhitelist [-u user] [-d duration] [package package-name]
  • 在使用者許可清單中新增/移除套件:
    cmd deviceidle whitelist [+/-]package-name
  • 檢查 jobscheduler 和鬧鐘管理工具的內部狀態:
    dumpsys jobscheduler
    dumpsys alarm

應用程式待命

應用程式待命功能會延遲使用者未積極使用的應用程式背景網路活動和工作,藉此延長電池續航力。

應用程式待命生命週期

平台會偵測閒置的應用程式,並將其置於應用程式待機狀態,直到使用者開始主動與應用程式互動為止。

偵測階段,平台會在裝置未充電時偵測應用程式是否處於非活動狀態,在特定時間內,使用者未直接或間接啟動應用程式,以及螢幕未開啟的特定時間。(當前景應用程式存取其他應用程式中的服務時,就會發生間接啟動情形)。

應用程式待命期間,平台會禁止應用程式一天內存取網路超過一次,並延後應用程式同步處理和其他工作。

平台會在下列情況下讓應用程式退出待命狀態

  • 應用程式會啟用。
  • 裝置已接上電源並開始充電。

處於活動狀態的應用程式不會受到應用程式待命功能的影響。應用程式「處於活動狀態」的條件如下:

  • 目前處於前景的程序 (做為活動或前景服務,或由其他活動或前景服務使用),例如通知事件監聽器、無障礙服務、動態桌布等。
  • 使用者查看的通知,例如在螢幕鎖定畫面或通知匣中
  • 已由使用者明確啟動

如果應用程式在一段時間內未發生上述任何活動,就會處於停用狀態。

測試應用程式待命功能

您可以使用下列 adb 指令手動測試應用程式待命功能:

adb shell dumpsys battery unplug
adb shell am set-idle package-name true
adb shell am set-idle package-name false
adb shell am get-idle package-name