如何在Doze模式Android 6.0中使用WhiteList应用程序

时间:2015-09-17 09:49:36

标签: java android preview android-6.0-marshmallow

此问题与Android 6.0预览版3有关,该版本将于本月底发布。

我正在使用Nexus 5' hammerhead'在Google的预览版3中测试Android 6.0中的一些内容。

新功能是"打盹模式" - 当网络被禁用且手机休眠时,类似深度睡眠模式,只有SMS,呼叫或高优先级GCM消息可以将其唤醒。但是像WhatsApp一样 - 在打盹模式下,它在2小时或更长时间后接收消息取决于计时器。但是有一个未经优化的列表'应用名为"白名单"你可以手动添加应用程序。

好的,我想找到一种方法,以编程方式添加我的应用程序而无需用户交互到白名单应用列表"在电池设置中存在于设备中。

尝试使用反射进入它我发现:

在android.os.IDeviceIdleController中有一个方法:

  

public abstract void addPowerSaveWhitelistApp(String packageNameOfApp)

但这是一个界面......所以我们不能创建一个接口实例。

目前还没有关于此接口或方法或任何继承树的文档。

也许您有一些想法,我应该在哪里寻找以编程方式添加我的应用程序的可能性?

还有一种方法

  

public abstract boolean isPowerSaveWhitelistApp(String packageName)

我觉得应该可以以某种方式访问​​?!检查应用程序是否存在于白名单中,也许最后希望ASK用户将其添加到白名单。

所以我的问题是,你们中的任何人都试图做出更好的结果吗?因为我被困住了,我认为这是一个死路一条。

了解更多信息:https://newcircle.com/s/post/1739/2015/06/12/diving-into-android-m-doze

6 个答案:

答案 0 :(得分:34)

如果没有Android M预览3上的用户互动,则无法禁用电池优化(=打盹模式的白名单应用程序)。

可以通过这种方式进行用户交互:

Intent intent = new Intent();
String packageName = context.getPackageName();
PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
if (pm.isIgnoringBatteryOptimizations(packageName))
    intent.setAction(Settings.ACTION_IGNORE_BATTERY_OPTIMIZATION_SETTINGS);
else {
    intent.setAction(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS);
    intent.setData(Uri.parse("package:" + packageName));
}
context.startActivity(intent);

答案 1 :(得分:34)

添加权限

<uses-permission 
android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS"/>

请求将您的应用列入白名单

 if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            Intent intent = new Intent();
            String packageName = getPackageName();
            PowerManager pm = (PowerManager) getSystemService(POWER_SERVICE);
            if (!pm.isIgnoringBatteryOptimizations(packageName)) {
                intent.setAction(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS);
                intent.setData(Uri.parse("package:" + packageName));
                startActivity(intent);
            }
        }

答案 2 :(得分:15)

我认为这个助手课程应该满足你的所有需求。

要使用它来请求操作系统将您的应用程序列入白名单,您可以使用prepareIntentForWhiteListingOfBatteryOptimization。如果你得到null,这意味着你不需要它,或者你不能使用它。您可以使用其他功能查询更好的状态。

public class PowerSaverHelper {
    public enum PowerSaveState {
        ON, OFF, ERROR_GETTING_STATE, IRRELEVANT_OLD_ANDROID_API
    }

    public enum WhiteListedInBatteryOptimizations {
        WHITE_LISTED, NOT_WHITE_LISTED, ERROR_GETTING_STATE, UNKNOWN_TOO_OLD_ANDROID_API_FOR_CHECKING, IRRELEVANT_OLD_ANDROID_API
    }

    public enum DozeState {
        NORMAL_INTERACTIVE, DOZE_TURNED_ON_IDLE, NORMAL_NON_INTERACTIVE, ERROR_GETTING_STATE, IRRELEVANT_OLD_ANDROID_API, UNKNOWN_TOO_OLD_ANDROID_API_FOR_CHECKING
    }

    @NonNull
    public static DozeState getDozeState(@NonNull Context context) {
        if (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP)
            return DozeState.IRRELEVANT_OLD_ANDROID_API;
        if (VERSION.SDK_INT < VERSION_CODES.M) {
            return DozeState.UNKNOWN_TOO_OLD_ANDROID_API_FOR_CHECKING;
        }
        final PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
        if (pm == null)
            return DozeState.ERROR_GETTING_STATE;
        return pm.isDeviceIdleMode() ? DozeState.DOZE_TURNED_ON_IDLE : pm.isInteractive() ? DozeState.NORMAL_INTERACTIVE : DozeState.NORMAL_NON_INTERACTIVE;
    }

    @NonNull
    public static PowerSaveState getPowerSaveState(@NonNull Context context) {
        if (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP)
            return PowerSaveState.IRRELEVANT_OLD_ANDROID_API;
        final PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
        if (pm == null)
            return PowerSaveState.ERROR_GETTING_STATE;
        return pm.isPowerSaveMode() ? PowerSaveState.ON : PowerSaveState.OFF;
    }


    @NonNull
    public static WhiteListedInBatteryOptimizations getIfAppIsWhiteListedFromBatteryOptimizations(@NonNull Context context, @NonNull String packageName) {
        if (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP)
            return WhiteListedInBatteryOptimizations.IRRELEVANT_OLD_ANDROID_API;
        if (VERSION.SDK_INT < VERSION_CODES.M)
            return WhiteListedInBatteryOptimizations.UNKNOWN_TOO_OLD_ANDROID_API_FOR_CHECKING;
        final PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
        if (pm == null)
            return WhiteListedInBatteryOptimizations.ERROR_GETTING_STATE;
        return pm.isIgnoringBatteryOptimizations(packageName) ? WhiteListedInBatteryOptimizations.WHITE_LISTED : WhiteListedInBatteryOptimizations.NOT_WHITE_LISTED;
    }

    @TargetApi(VERSION_CODES.M)
    @RequiresPermission(permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS)
    @Nullable
    public static Intent prepareIntentForWhiteListingOfBatteryOptimization(@NonNull Context context, @NonNull String packageName, boolean alsoWhenWhiteListed) {
        if (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP)
            return null;
        if (ContextCompat.checkSelfPermission(context, permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS) == PackageManager.PERMISSION_DENIED)
            return null;
        final WhiteListedInBatteryOptimizations appIsWhiteListedFromPowerSave = getIfAppIsWhiteListedFromBatteryOptimizations(context, packageName);
        Intent intent = null;
        switch (appIsWhiteListedFromPowerSave) {
            case WHITE_LISTED:
                if (alsoWhenWhiteListed)
                    intent = new Intent(Settings.ACTION_IGNORE_BATTERY_OPTIMIZATION_SETTINGS);
                break;
            case NOT_WHITE_LISTED:
                intent = new Intent(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS).setData(Uri.parse("package:" + packageName));
                break;
            case ERROR_GETTING_STATE:
            case UNKNOWN_TOO_OLD_ANDROID_API_FOR_CHECKING:
            case IRRELEVANT_OLD_ANDROID_API:
            default:
                break;
        }
        return intent;
    }

    /**
     * registers a receiver to listen to power-save events. returns true iff succeeded to register the broadcastReceiver.
     */
    @TargetApi(VERSION_CODES.M)
    public static boolean registerPowerSaveReceiver(@NonNull Context context, @NonNull BroadcastReceiver receiver) {
        if (VERSION.SDK_INT < VERSION_CODES.M)
            return false;
        IntentFilter filter = new IntentFilter();
        filter.addAction(PowerManager.ACTION_DEVICE_IDLE_MODE_CHANGED);
        context.registerReceiver(receiver, filter);
        return true;
    }


}

答案 3 :(得分:4)

据我所知,你不能将自己列入打瞌睡的白名单。忽略电池优化不会禁用打盹。见这里:https://code.google.com/p/android-developer-preview/issues/detail?id=2225 更新:在M的发布版本中,您可以请求忽略电池优化,这至少可以让您在打盹模式下正常访问互联网。

答案 4 :(得分:3)

更新

当设备再次静止,屏幕关闭并且电池供电一段时间后,Doze会对 PowerManager.WakeLock,AlarmManager警报和GPS / Wi-Fi扫描应用完整的CPU和网络限制

访问Use Cases for Whitelisting了解更多详情。

  

下表突出显示了请求或的可接受用例   正在使用电池优化例外白名单。一般来说,   除非Doze或App Standby,否则您的应用不应位于白名单中   打破应用程序的核心功能或有一个技术原因   您的应用无法使用GCM高优先级消息。

android n developer

Doze特别可能影响AlarmManager 警报计时器管理的活动,因为 Android 5.1(API级别22)或更低级别的警报不会触发该系统处于Doze

Android 6.0(API级别23)引入了两种新的AlarmManager方法:setAndAllowWhileIdle()setExactAndAllowWhileIdle()。使用这些方法,您可以设置即使设备处于打盹状态也会触发的警报。

注意setAndAllowWhileIdle()setExactAndAllowWhileIdle()都不能每个应用每15分钟触发一次警报。

Testing with Doze and App Standby

答案 5 :(得分:3)

如果我们拥有root权限,则可以以编程方式完成白名单。 要在白名单应用程序中添加设备,请从应用程序运行以下命令:

adb shell dumpsys deviceidle whitelist +<package_name>

要从白名单应用程序中删除设备,请从应用程序运行以下命令:

adb shell dumpsys deviceidle whitelist -<package_name>