ใน Android 9 ขึ้นไป แพลตฟอร์มจะตรวจสอบแอปเพื่อหาลักษณะการทำงานที่ส่งผลเสียต่ออายุการใช้งานแบตเตอรี่ของอุปกรณ์ แพลตฟอร์มจะใช้และประเมินกฎการตั้งค่าเพื่อให้โฟลว์ UX ที่เปิดโอกาสให้ผู้ใช้จำกัดแอปที่ละเมิดกฎ
ใน Android 8.0 และต่ำกว่า มีการจำกัดผ่านฟีเจอร์ต่างๆ เช่น โหมดสลีป แอปรอใช้งาน ขีดจำกัดของเบื้องหลัง และขีดจำกัดตำแหน่งในเบื้องหลัง อย่างไรก็ตาม แอปบางแอปยังคงแสดงลักษณะการทำงานที่ไม่ถูกต้องอยู่ ซึ่งแอปบางแอปมีคำอธิบายอยู่ใน Android Vitals Android 9 ได้เปิดตัวโครงสร้างพื้นฐานของระบบปฏิบัติการที่สามารถตรวจหาและจํากัดแอปตามกฎการตั้งค่าที่อัปเดตได้เมื่อเวลาผ่านไป
ข้อจำกัดการทำงานในเบื้องหลัง
ผู้ใช้สามารถจำกัดแอป หรือระบบอาจแนะนำแอปที่ตรวจพบว่าส่งผลเสียต่อประสิทธิภาพของอุปกรณ์
แอปที่ถูกจำกัด
- ผู้ใช้ยังคงเปิดแอปได้
- เรียกใช้งาน/การปลุก หรือใช้เครือข่ายในเบื้องหลังไม่ได้
- เรียกใช้บริการที่ทำงานอยู่เบื้องหน้าไม่ได้
- ผู้ใช้เปลี่ยนเป็นแอปที่ไม่มีข้อจำกัดได้
ผู้ติดตั้งใช้งานอุปกรณ์สามารถเพิ่มข้อจำกัดเพิ่มเติมลงในแอปเพื่อดำเนินการต่อไปนี้
- จำกัดไม่ให้แอปรีสตาร์ทเอง
- จำกัดไม่ให้บริการมีการเชื่อมโยง (มีความเสี่ยงสูง)
แอปที่ถูกจำกัดในเบื้องหลังไม่ควรใช้ทรัพยากรของอุปกรณ์ เช่น หน่วยความจำ, CPU และแบตเตอรี่ แอปที่ถูกจำกัดการทำงานในเบื้องหลังไม่ควรส่งผลต่อประสิทธิภาพของอุปกรณ์เมื่อผู้ใช้ไม่ได้ใช้งานแอปเหล่านั้น อย่างไรก็ตาม แอปเดียวกันนี้ควรทำงานได้อย่างเต็มรูปแบบเมื่อผู้ใช้เปิดแอป
ใช้การติดตั้งใช้งานที่กําหนดเอง
ผู้ติดตั้งใช้งานอุปกรณ์จะใช้วิธีการที่กำหนดเองเพื่อใช้ข้อจำกัดกับแอปต่อไปได้
ผสานรวมข้อจำกัดของแอป
ส่วนต่อไปนี้จะอธิบายวิธีกำหนดและผสานรวมการจำกัดแอปในอุปกรณ์ หากคุณใช้วิธีการจํากัดแอปจาก Android 8.x หรือต่ำกว่า โปรดอ่านส่วนต่อไปนี้อย่างละเอียดเพื่อดูการเปลี่ยนแปลงใน Android 9 ขึ้นไป
ตั้งค่า Flag ของ AppOpsManager
เมื่อมีการจํากัดแอป ให้ตั้งค่า Flag ที่เหมาะสมใน 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 แสดงผลเป็น "จริง"
เมื่อแอปถูกจํากัด ให้ตรวจสอบว่า 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
สแตนด์บายแอป
สถานะ "รอดำเนินการ" ของแอปจะช่วยยืดอายุการใช้งานแบตเตอรี่โดยการเลื่อนกิจกรรมและงานในเครือข่ายเบื้องหลังสำหรับแอปที่ผู้ใช้ไม่ได้ใช้งานอยู่
วงจรสแตนด์บายของแอป
แพลตฟอร์มจะตรวจหาแอปที่ไม่ได้ใช้งานและวางแอปเหล่านั้นไว้ในโหมดสแตนด์บายของแอปจนกว่าผู้ใช้จะเริ่มมีส่วนร่วมกับแอปอย่างจริงจัง
ในระหว่างระยะการตรวจจับ แพลตฟอร์มจะตรวจพบว่าแอปไม่มีการใช้งานเมื่ออุปกรณ์ไม่ได้ชาร์จ และผู้ใช้ไม่ได้เปิดแอปโดยตรงหรือโดยอ้อมเป็นระยะเวลาหนึ่งๆ รวมถึงไม่ได้เปิดหน้าจอเป็นระยะเวลาหนึ่งๆ (การเปิดใช้งานโดยอ้อมเกิดขึ้นเมื่อแอปที่ทำงานอยู่เบื้องหน้าเข้าถึงบริการในแอปที่ 2)
ในระหว่างโหมดรอของแอป แพลตฟอร์มจะป้องกันไม่ให้แอปเข้าถึงเครือข่ายมากกว่าวันละครั้ง โดยจะเลื่อนเวลาการซิงค์แอปและงานอื่นๆ
แพลตฟอร์มจะออกจากโหมดสแตนด์บายของแอปในกรณีต่อไปนี้
- แอปจะทำงาน
- อุปกรณ์เสียบปลั๊กและชาร์จอยู่
แอปที่ใช้งานอยู่จะไม่ได้รับผลกระทบจากการรอแอป แอปจะทำงานอยู่เมื่อมีลักษณะดังนี้
- กระบวนการที่ทำงานอยู่เบื้องหน้า (เป็นกิจกรรมหรือบริการที่ทำงานอยู่เบื้องหน้า หรือมีการใช้งานโดยกิจกรรมหรือบริการที่ทำงานอยู่เบื้องหน้าอื่น) เช่น โปรแกรมรับฟังการแจ้งเตือน บริการการช่วยเหลือพิเศษ วอลเปเปอร์เคลื่อนไหว ฯลฯ
- การแจ้งเตือนที่ผู้ใช้ดู เช่น ในหน้าจอล็อกหรือถาดการแจ้งเตือน
- ผู้ใช้เปิดใช้งานอย่างชัดเจน
แอปจะไม่ทำงานหากไม่มีกิจกรรมข้างต้นเกิดขึ้นเป็นระยะเวลาหนึ่ง
ทดสอบโหมดสแตนด์บายของแอป
คุณสามารถทดสอบโหมด standby ของแอปด้วยตนเองได้โดยใช้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