此问题与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
答案 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高优先级消息。
Doze特别可能影响AlarmManager 警报和计时器管理的活动,因为 Android 5.1(API级别22)或更低级别的警报不会触发该系统处于Doze 。
Android 6.0(API级别23)引入了两种新的AlarmManager方法:setAndAllowWhileIdle()
和setExactAndAllowWhileIdle()
。使用这些方法,您可以设置即使设备处于打盹状态也会触发的警报。
注意:setAndAllowWhileIdle()
和setExactAndAllowWhileIdle()
都不能每个应用每15分钟触发一次警报。
答案 5 :(得分:3)
如果我们拥有root权限,则可以以编程方式完成白名单。 要在白名单应用程序中添加设备,请从应用程序运行以下命令:
adb shell dumpsys deviceidle whitelist +<package_name>
要从白名单应用程序中删除设备,请从应用程序运行以下命令:
adb shell dumpsys deviceidle whitelist -<package_name>