در اندروید 9 و بالاتر، این پلتفرم میتواند برنامهها را از نظر رفتاری که بر عمر باتری دستگاهها تأثیر منفی میگذارد نظارت کند. این پلتفرم از قوانین راه اندازی استفاده و ارزیابی می کند تا یک جریان UX ارائه دهد که به کاربران این امکان را می دهد تا برنامه هایی را که قوانین را نقض می کنند محدود کنند.
در اندروید 8.0 و پایینتر، محدودیتهایی از طریق ویژگیهایی مانند Doze، آماده به کار برنامه، محدودیتهای پسزمینه و محدودیتهای موقعیت مکانی پسزمینه وجود داشت. با این حال، برخی از برنامهها همچنان رفتارهای بدی از خود نشان میدهند، که برخی از آنها در Android vitals توضیح داده شدهاند. اندروید 9 زیرساخت سیستم عاملی را معرفی کرد که میتواند برنامهها را بر اساس قوانین راهاندازی شناسایی و محدود کند که میتوانند در طول زمان بهروزرسانی شوند.
محدودیت های پس زمینه
کاربران میتوانند برنامهها را محدود کنند یا سیستم ممکن است برنامههایی را پیشنهاد دهد که تشخیص میدهد بر سلامت دستگاه تأثیر منفی میگذارند.
برنامه های محدود شده:
- هنوز هم کاربر می تواند راه اندازی شود.
- نمیتوان مشاغل/زنگها را اجرا کرد یا از شبکه در پسزمینه استفاده کرد.
- نمی توان سرویس های پیش زمینه را اجرا کرد.
- می تواند توسط کاربر به یک برنامه نامحدود تغییر یابد.
پیادهکنندههای دستگاه میتوانند محدودیتهای اضافی را به برنامهها اضافه کنند تا:
- برنامه را از راه اندازی مجدد خود محدود کنید.
- محدود کردن خدمات از محدود بودن (بسیار خطرناک).
برنامههای محدود شده در پسزمینه انتظار نمیرود منابع دستگاهی مانند حافظه، CPU و باتری را مصرف کنند. وقتی کاربر به طور فعال از آن برنامهها استفاده نمیکند، برنامههای دارای محدودیت پسزمینه نباید بر سلامت دستگاه تأثیر بگذارند. با این حال، انتظار می رود زمانی که کاربر برنامه ها را راه اندازی می کند، همان برنامه ها کاملاً کاربردی باشند.
از پیاده سازی های سفارشی استفاده کنید
پیادهکنندههای دستگاه میتوانند به استفاده از روشهای سفارشی خود برای اعمال محدودیتها بر روی برنامهها ادامه دهند.
ادغام محدودیت های برنامه
بخشهای زیر نحوه تعریف و ادغام محدودیتهای برنامه را در دستگاه خود شرح میدهند. اگر از روشهای محدودیت برنامه از اندروید 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 درست است
وقتی یک برنامه محدود می شود، مطمئن شوید که 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)); }
مقدار AnomalyType
را جایگزین type
کنید.
پیادهکنندههای دستگاه میتوانند از ثابتهای تعریفشده در 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)); }
تست محدودیت های اپلیکیشن
برای تست رفتار محدودیت های برنامه در اندروید 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