使用打盹模式在前台服务中禁用唤醒锁定 - Android M中的新电池优化

Pet*_*vka 11 android android-6.0-marshmallow

如果来自Dianne Hackborn的关于G +帖子的评论构成"记录":https://plus.google.com/+AndroidDevelopers/posts/94jCkmG4jff但是应该允许与通知图标关联的前台服务,这显然是半记录的行为在打盹模式下保持唤醒锁定.

除了前台服务之外,如果您有顶级活动,这似乎无效.

我创建了一个最小的实现,演示了这个:https: //github.com/petrnalevka/dozetest/blob/master/src/com/urbandroid/doze/DozeService.java

问题可以在
Android M,MRA58K Nexus 5和MRA58N Nexus 6上重现

我的前台服务有一个相关的通知,我持有部分唤醒锁.不幸的是,打盹模式接管了我的唤醒锁定.我只能通过退出电池优化或退出最高活动来阻止这种情况.

我相信这是Android中的一个错误,因为没有理由为什么前台服务应该保持唤醒锁定,但如果他们在顶部有活动则没有.

即使我已经在Androdi问题跟踪器上报告了这个问题,我也已将此问题报告在此处https://code.google.com/p/android/issues/detail?id=193802,以便找到解决方法,因为这是一个重要的问题功能,以便不需要REQUEST_IGNORE_BATTERY_OPTIMIZATIONS,这显然只是意味着REQUEST_REMOVAL_FROM_PLAYSTORE_BY_GOOGLE.

以下是我使用adb重现问题的方法,感谢+ Dianne Hackborn的提示:

似乎当我将一个Activity保留在前台,同时还有运行我的应用程序的前台服务时,Power Manager会将其识别为:

PARTIAL_WAKE_LOCK              'Doze lock' DISABLED (uid=10112, pid=24649, ws=null)  
Proc # 0: fore  F/A/TS trm: 0 24649:com.urbandroid.doze/u0a112 (top-activity)  
Run Code Online (Sandbox Code Playgroud)

如果我按回家并离开活动,我会进入州:

PARTIAL_WAKE_LOCK              'Doze lock' (uid=10112, pid=24649, ws=null)  
Proc # 4: prcp  F/S/SF trm: 0 24649:com.urbandroid.doze/u0a112 (fg-service)  
Run Code Online (Sandbox Code Playgroud)

Pet*_*vka 8

这是Dianne Hackborn在https://plus.google.com/+AndroidDevelopers/posts/94jCkmG4jff下发布的答案.我们仍然可以在这里集思广益,因为分离流程可能并不总是直截了当.

Dianne Hackborn:+ Petr Nalevka很抱歉,这似乎是平台上的一个错误.我会尽可能地修复它.

你的解决方法是正确的方向,但请不要做那种在屏幕上打开/关闭你的状态的事情.好吧,当屏幕关闭时可以向后移动并做到这一点...但是,当屏幕打开时强迫自己到前面是非常错误的,因为它可能因其他原因而打开(到发射相机,语音交互等).这实际上可能是平台应该更好地保护自己.

我建议的是听打瞌睡模式开始,并在这种情况下将你的活动放在后面.您可以通过收听广播http://developer.android.com/reference/android/os/PowerManager.html#ACTION_DEVICE_IDLE_MODE_CHANGED并在看到设备空闲为真时将自己发送到后面来执行此操作.我建议不要担心试图让自己回到前台 - 在用户实际上很长一段时间没有使用他们的设备并且发生这种情况的情况下,回到它并且在家里不应该真是个惊喜.

EDIT(作者:Dianne Hackborn):实际上,另一种解决方案 - 让您的前台服务在与活动不同的过程中运行.从我所看到的,这将工作正常.如果你在那里得到理想的行为,我会很有趣.

此外,这实际上是我们针对这种情况的推荐做法 - 如果您有一个长期运行的前台服务,它应该在与活动不同的进程中,因此它不会强制保留与该活动相关的所有内存周围.(这也是这个bug通过的原因,我们所有的应用程序都使用这种模式.)