Android Doze低电耗休眠模式 与 WorkManager详解
<div id="navCategory"><h5 class="catalogue">目录</h5><ul class="first_class_ul"><li>1. Doze模式下,WorkManager setInitialDelay设置小于15分钟,被系统强制到15分钟执行,怎么办 ?</li><li>2. 了解低电耗模式(Doze)</li><li>3. 低电耗模式限制</li><li>4. Doze模式下,WorkManager 为何无法精确时间执行 ?</li><li>5. Doze模式下,如何精确时间执行 ?</li><ul class="second_class_ul"><li>5.1 声明权限</li><li>5.2 调用前需验证是否已授权</li><li>5.3 创建 BroadcastReceiver : 接收闹钟触发事件</li><li>5.4 设置精确闹钟</li><li>5.5 设置了AlarmManager,如果系统时间变更了,是不是闹钟在现实世界响起的时间也会变 ?</li><li>5.6 STATE_DOZE和STATE_DOZE_SUSPEND,有什么区别 ?</li></ul><li>6. STATE_DOZE 和 STATE_DOZE_SUSPEND</li><ul class="second_class_ul"><li>6.1 CPU 与任务执行机制**</li><li>6.2 网络与后台资源访问</li><li>6.3 闹钟与任务调度行为</li><li>6.4 Wakelock 处理</li><li>6.5 持续时间与退出机制</li><li>6.6 对比总结</li><li>6.7 适配建议</li></ul><li>7. STATE_DOZE_SUSPEND状态下设置setExactAndAllowWhileIdle有效吗</li><ul class="second_class_ul"><li>7.1STATE_DOZE_SUSPEND的特性</li><li>7.2setExactAndAllowWhileIdle()的限制</li><li>7.3 替代方案</li><li>7.4 总结:不同闹钟方法在 Doze 模式下的有效性</li></ul><li>8. 更多内容</li><ul class="second_class_ul"></ul></ul></div><p class="maodian"></p><h2>1. Doze模式下,WorkManager setInitialDelay设置小于15分钟,被系统强制到15分钟执行,怎么办 ?</h2><p>Android 拥有两项省电功能,通过管理设备未连接电源时应用的行为来延长用户电池续航时间:低电耗模式 (Doze) 和应用待机模式 (App Standby)。 低电耗模式 通过延迟设备长时间未使用时应用的后台 CPU 和网络活动来降低电池消耗。 应用待机模式 延迟没有近期用户活动的应用的后台网络活动。</p>
<p>设备处于低电耗模式时,应用对某些耗电资源的访问会被延迟,直到维护窗口。 具体限制列在电量管理限制中。</p>
<p>低电耗模式和应用待机模式管理在 Android 6.0 或更高版本上运行的所有应用的行为,无论它们是否专门针对 API 级别 23。为了帮助确保为用户提供最佳体验,请在低电耗模式和应用待机模式下测试您的应用,并对您的代码进行任何必要的调整。以下部分提供详细信息。</p>
<p class="maodian"></p><h2>2. 了解低电耗模式(Doze)</h2>
<p>如果用户将设备拔下电源并长时间静置,且屏幕关闭,设备就会进入低电耗模式。在低电耗模式下,系统会尝试通过限制应用对网络和 CPU 密集型服务的访问来节省电池电量。它还会阻止应用访问网络,并延迟其作业、同步和标准闹钟。</p>
<p>系统会定期短暂退出低电耗模式,让应用完成其延迟的活动。在此 维护窗口 期间,系统会运行所有待处理的同步、作业和闹钟,并允许应用访问网络。</p>
<p style="text-align:center"><img alt="" src="https://img.jbzj.com/file_images/article/202509/2025092014515662.png" /></p>
<p>维护窗口结束后,系统会再次进入低电耗模式,暂停网络访问并延迟作业、同步和闹钟。随着时间的推移,系统安排维护窗口的频率会降低,有助于在设备未充电且长时间不活动的情况下减少电池消耗。</p>
<p>当用户通过移动设备、打开屏幕或连接充电器来唤醒设备时,系统会退出低电耗模式,所有应用都会恢复正常活动。</p>
<p class="maodian"></p><h2>3. 低电耗模式限制</h2>
<p>设备处于低电耗模式时,系统会对您的应用施加以下限制:</p>
<ul><li>暂停网络访问。</li><li>忽略唤醒锁。</li><li>将标准 AlarmManager 闹钟(包括 setExact() 和 setWindow())延迟到下一个维护窗口。<ul><li>如果您需要在低电耗模式下触发闹钟,请使用 setAndAllowWhileIdle() 或 setExactAndAllowWhileIdle()。</li><li>通过 setAlarmClock() 设置的闹钟会正常触发。系统会在这些闹钟触发前不久退出低电耗模式。</li></ul></li><li>不执行 Wi-Fi 扫描。</li><li>不允许同步适配器运行。</li><li>不允许 JobScheduler 运行。</li></ul>
<blockquote><p>⚠️ WorkManager 内部使用 JobScheduler,因此 WorkManager 任务不会运行。</p></blockquote>
<blockquote><p>具体详见 Android Developer | doze-standby</p></blockquote>
<p class="maodian"></p><h2>4. Doze模式下,WorkManager 为何无法精确时间执行 ?</h2>
<p>通过上文,我们可以知道,在Doze模式下,WorkManager 任务不会运行,只有在两个Doze间隔期间,系统会定期短暂退出低电耗模式,让应用完成其延迟的活动。在此 维护窗口 期间,系统会运行所有待处理的同步、作业和闹钟,并允许应用访问网络。 这个时候WorkManager的任务才会被执行。但是两个Doze之间休眠时间的间隔是不确定的,所以Doze模式下,WorkManager无法精确时间被执行。</p>
<p>且如果WorkManager.setInitialDelay设置的时间小于15分钟,会被系统强制替换为15分钟。<br />那如果我就想10分钟后执行,需要怎么办呢 ?</p>
<p class="maodian"></p><h2>5. Doze模式下,如何精确时间执行 ?</h2>
<p>这个时候,就需要使用AlarmManager的setExactAndAllowWhileIdle方法了。 (虽然WorkManager在Android低版本上也是用的AlarmManager,但是并没有使用AlarmManager的setExactAndAllowWhileIdle方法)</p>
<p>那么如何使用呢 ?</p>
<p class="maodian"></p><h3>5.1 声明权限</h3>
<div class="jb51code"><pre class="brush:java;"><uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM" /></pre></div>
<p class="maodian"></p><h3>5.2 调用前需验证是否已授权</h3>
<div class="jb51code"><pre class="brush:java;">if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) {
AlarmManager alarmManager = (AlarmManager) getSystemService(ALARM_SERVICE);
if (!alarmManager.canScheduleExactAlarms()) {
// 引导用户前往设置页授权
Intent intent = new Intent(Settings.ACTION_REQUEST_SCHEDULE_EXACT_ALARM);
intent.setData(Uri.parse("package:" + getPackageName()));
startActivity(intent);
return;
}
}</pre></div>
<p class="maodian"></p><h3>5.3 创建 BroadcastReceiver : 接收闹钟触发事件</h3>
<div class="jb51code"><pre class="brush:java;">public class AlarmReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
// 执行任务逻辑(如启动服务、发送通知)
Log.d("Alarm", "Triggered at exact time!");
}
}</pre></div>
<p>别忘了注册 Receiver</p>
<div class="jb51code"><pre class="brush:java;"><receiver android:name=".AlarmReceiver" android:exported="false"/></pre></div>
<p class="maodian"></p><h3>5.4 设置精确闹钟</h3>
<div class="jb51code"><pre class="brush:java;">// 获取 AlarmManager
AlarmManager alarmManager = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
// 创建 Intent 指向 BroadcastReceiver
Intent intent = new Intent(this, AlarmReceiver.class);
PendingIntent pendingIntent = PendingIntent.getBroadcast(
this, 0, intent,
PendingIntent.FLAG_UPDATE_CURRENT | PendingIntent.FLAG_IMMUTABLE
);
// 设置触发时间(例如 10 分钟后)
long triggerTime = System.currentTimeMillis() + 10 * 60 * 1000;
// 根据版本选择方法
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
alarmManager.setExactAndAllowWhileIdle(
AlarmManager.RTC_WAKEUP, // 使用 UTC 时间并唤醒设备
triggerTime,
pendingIntent
);
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
alarmManager.setExact(AlarmManager.RTC_WAKEUP, triggerTime, pendingIntent);
} else {
alarmManager.set(AlarmManager.RTC_WAKEUP, triggerTime, pendingIntent);
}</pre></div>
<p class="maodian"></p><h3>5.5 设置了AlarmManager,如果系统时间变更了,是不是闹钟在现实世界响起的时间也会变 ?</h3>
<ul><li>若用户将系统时间调快 1 小时(从 9:00 改为 10:00),闹钟会立即触发(因为系统时间已达到目标值)。</li><li>若将系统时间调慢 1 小时(从 10:00 改为 9:00),闹钟会延迟 1 小时触发(需等待系统时间再次达到 10:00)</li></ul>
<p>可以监听系统时间变化的广播,然后更改闹钟</p>
<div class="jb51code"><pre class="brush:java;">// 在 onResume() 中注册
IntentFilter filter = new IntentFilter();
filter.addAction(Intent.ACTION_TIME_CHANGED);
filter.addAction(Intent.ACTION_TIMEZONE_CHANGED);
registerReceiver(timeChangeReceiver, filter);
// 在 onPause() 中注销
unregisterReceiver(timeChangeReceiver);</pre></div>
<p class="maodian"></p><h3>5.6 STATE_DOZE和STATE_DOZE_SUSPEND,有什么区别 ?</h3>
<p>在 Android 电源管理机制中,<code>STATE_DOZE</code> 和 <code>STATE_DOZE_SUSPEND</code> 是两种不同的休眠状态,其核心区别在于<strong>系统资源限制的严格程度</strong>和<strong>CPU活动状态</strong>。以下是两者的详细对比:</p>
<p class="maodian"></p><h2>6. STATE_DOZE 和 STATE_DOZE_SUSPEND</h2>
<ul><li><code>STATE_DOZE</code>(Doze 模式)
<ul><li>首次引入于 Android 6.0(Marshmallow),目的是在设备闲置时(未充电、屏幕关闭、静止状态)减少后台活动。</li><li>状态描述:系统进入低功耗状态,但CPU仍保持部分活动,周期性唤醒处理任务。</li></ul></li><li><code>STATE_DOZE_SUSPEND</code>(Doze 挂起模式)<ul><li>强化版省电机制,于 Android 9.0(Pie) 引入,作为 Doze 的深度扩展。</li><li>状态描述:系统进入更深度的休眠,CPU 完全停止运行,仅保留最低限度的硬件唤醒能力(如传感器)。</li></ul></li></ul>
<p class="maodian"></p><h3>6.1 CPU 与任务执行机制**</h3>
<ul><li><code>STATE_DOZE</code>:
<ul><li>CPU 未完全停止,而是周期性唤醒(维护窗口)。</li><li>维护窗口间隔:初始为每 30 分钟唤醒一次,随后间隔逐渐延长(如 1 小时、2 小时)。</li><li>任务处理:在维护窗口内,系统允许应用执行延迟的任务(如同步、AlarmManager 闹钟)。</li></ul></li><li><code>STATE_DOZE_SUSPEND</code>:<ul><li>CPU 完全挂起,无周期性唤醒。</li><li>任务冻结:所有后台进程被强制暂停(通过 Linux cgroup 的 <code>freezer</code> 子系统),不再分配 CPU 时间片。</li><li>唤醒条件:仅通过外部事件触发(如用户操作、高优先级闹钟 <code>setAlarmClock()</code>)。</li></ul></li></ul>
<p class="maodian"></p><h3>6.2 网络与后台资源访问</h3>
<ul><li><code>STATE_DOZE</code>:
<ul><li>网络限制:禁止后台应用访问网络,仅维护窗口内开放。</li><li>部分豁免:高优先级 GCM 消息、短信/电话可临时唤醒网络。</li></ul></li><li><code>STATE_DOZE_SUSPEND</code>:<ul><li>完全断网:所有网络访问被禁止,包括 GCM 和短信(仅保留基础通信服务如通话)。</li><li>硬件限制:Wi-Fi/GPS 扫描、传感器数据采集均暂停。</li></ul></li></ul>
<p class="maodian"></p><h3>6.3 闹钟与任务调度行为</h3>
<ul><li><code>STATE_DOZE</code>:
<ul><li>标准闹钟延迟:<code>setExact()</code> 或 <code>setWindow()</code> 的闹钟被推迟至下一个维护窗口。</li><li>豁免闹钟:<code>setAndAllowWhileIdle()</code> 或 <code>setExactAndAllowWhileIdle()</code> 可在 Doze 下触发(但受每分钟 1 次的频率限制)。</li></ul></li><li><code>STATE_DOZE_SUSPEND</code>:<ul><li>所有闹钟冻结:包括 <code>AllowWhileIdle</code> 类型的闹钟,仅 <code>setAlarmClock()</code>(用户可见的闹钟)可唤醒设备。</li><li>任务调度失效:JobScheduler 和 WorkManager 任务被无限期推迟,直至退出 SUSPEND 状态。</li></ul></li></ul>
<p class="maodian"></p><h3>6.4 Wakelock 处理</h3>
<ul><li><code>STATE_DOZE</code>:
<ul><li>部分屏蔽:普通 WakeLock 被忽略(如 <code>PARTIAL_WAKE_LOCK</code>),但高优先级服务(如媒体播放)可能被豁免。</li></ul></li><li><code>STATE_DOZE_SUSPEND</code>:<ul><li>完全无效:所有 WakeLock 被强制释放,无法阻止 CPU 挂起。</li></ul></li></ul>
<p class="maodian"></p><h3>6.5 持续时间与退出机制</h3>
<ul><li><code>STATE_DOZE</code>:
<ul><li>动态维护窗口:窗口间隔随闲置时间延长而增加(30 分钟 → 1 小时 → 数小时)。</li><li>退出条件:屏幕点亮、设备移动或充电。</li></ul></li><li><code>STATE_DOZE_SUSPEND</code>:<ul><li>持续挂起:无周期性窗口,直至外部事件唤醒。</li><li>退出条件更严格:仅用户交互(如按键)、<code>setAlarmClock()</code> 闹钟或充电可唤醒。</li></ul></li></ul>
<p class="maodian"></p><h3>6.6 对比总结</h3>
<table><thead><tr><th><strong>特性</strong></th><th><strong>STATE_DOZE</strong></th><th><strong>STATE_DOZE_SUSPEND</strong></th></tr></thead><tbody><tr><td><strong>CPU 状态</strong></td><td>周期性唤醒(维护窗口)</td><td>完全停止</td></tr><tr><td><strong>网络访问</strong></td><td>仅维护窗口开放</td><td>完全禁止</td></tr><tr><td><strong>闹钟执行</strong></td><td>允许 <code>AllowWhileIdle</code> 类型</td><td>仅 <code>setAlarmClock()</code> 有效</td></tr><tr><td><strong>任务调度</strong></td><td>维护窗口内执行</td><td>完全冻结</td></tr><tr><td><strong>Wakelock 有效性</strong></td><td>部分豁免</td><td>全部释放</td></tr><tr><td><strong>退出条件</strong></td><td>移动/亮屏/充电</td><td>仅用户操作或高优先级闹钟</td></tr><tr><td><strong>省电强度</strong></td><td>中度</td><td>极端</td></tr></tbody></table>
<p class="maodian"></p><h3>6.7 适配建议</h3>
<ol><li><strong>后台任务</strong>:
<ul><li>使用 <code>WorkManager</code> 或 <code>JobScheduler</code>,其任务在 <code>STATE_DOZE</code> 的维护窗口自动执行,但 <code>STATE_DOZE_SUSPEND</code> 中仍会被冻结。</li></ul></li><li><strong>实时性需求</strong>:<ul><li>关键任务(如闹钟)优先用 <code>setAlarmClock()</code>(用户可见),或引导用户将应用加入电池优化白名单。</li></ul></li><li><strong>网络依赖</strong>:<ul><li>在 <code>STATE_DOZE_SUSPEND</code> 下网络完全不可用,需通过高优先级 GCM 消息或短信唤醒设备(系统服务如电话/SMS 始终豁免)。</li></ul></li><li><strong>状态检测</strong>:<ul><li>通过 <code>PowerManager.isDeviceIdleMode()</code> 检测 Doze 状态,但无法直接区分 <code>SUSPEND</code>(需依赖行为推断)。</li></ul></li></ol>
<blockquote><p>📌 <strong>核心结论</strong>:<br /><code>STATE_DOZE</code> 是 <strong>周期性唤醒的轻度休眠</strong>,适合延迟容忍的任务;<br /><code>STATE_DOZE_SUSPEND</code> 是 <strong>CPU 完全停止的深度冻结</strong>,仅响应极高优先级事件。<br />开发者需针对两者设计降级策略,确保关键功能在极端省电下仍可工作。</p></blockquote>
<p class="maodian"></p><h2>7. STATE_DOZE_SUSPEND状态下设置setExactAndAllowWhileIdle有效吗</h2>
<p>在 Android 的 <code>STATE_DOZE_SUSPEND</code>(深度挂起模式)状态下,<strong><code>setExactAndAllowWhileIdle()</code> 是无效的</strong>。以下是具体原因和机制分析:</p>
<p class="maodian"></p><h3>7.1STATE_DOZE_SUSPEND的特性</h3>
<ul><li><strong>CPU 完全冻结</strong>:<br />此状态下系统进入深度休眠,<strong>CPU 完全停止运行</strong>,所有后台进程被强制暂停,仅保留基础硬件唤醒能力(如高优先级闹钟或物理按键)。</li><li><strong>网络与任务冻结</strong>:<br />所有网络访问被禁止,<code>JobScheduler</code>、<code>WorkManager</code> 等后台任务被无限期推迟,标准闹钟和 <code>AllowWhileIdle</code> 类闹钟均被冻结。</li></ul>
<p class="maodian"></p><h3>7.2setExactAndAllowWhileIdle()的限制</h3>
<ul><li><strong>仅适用于普通 Doze 模式</strong>:<br /><code>setExactAndAllowWhileIdle()</code> 设计用于 <code>STATE_DOZE</code>(普通休眠),可在维护窗口外触发(但受每分钟 1 次的频率限制)。但在 <code>STATE_DOZE_SUSPEND</code> 下,<strong>系统无周期性唤醒机制</strong>,导致该方法失效。</li><li><strong>深度省电下的唤醒条件</strong>:<br />仅以下事件可唤醒 <code>STATE_DOZE_SUSPEND</code>:<ul><li>用户主动操作(如点亮屏幕)</li><li>高优先级闹钟 <strong><code>setAlarmClock()</code></strong>(系统会提前退出休眠并显示通知)</li><li>设备充电。</li></ul></li></ul>
<p class="maodian"></p><h3>7.3 替代方案</h3>
<p>若需在 <code>STATE_DOZE_SUSPEND</code> 下可靠触发任务,需采用以下策略:</p>
<ol><li><code>setAlarmClock()</code>:
<ul><li>用于用户可见的精确闹钟(如闹钟应用),系统会强制退出休眠并显示通知栏图标。</li><li>无需特殊权限,但用户感知明显。</li></ul></li><li><strong>引导用户加入白名单</strong>:<ul><li>用户手动在 <strong>设置→电池优化</strong> 中将应用设为“未优化”,可部分豁免限制(非完全保障,厂商兼容性差异大)。</li></ul></li><li><strong>前台服务(Foreground Service)</strong>:<ul><li>通过持续通知栏服务维持进程活跃性,但需合理说明用途以避免被系统限制或用户关闭。</li></ul></li></ol>
<p class="maodian"></p><h3>7.4 总结:不同闹钟方法在 Doze 模式下的有效性</h3>
<table><thead><tr><th><strong>闹钟方法</strong></th><th><code>STATE_DOZE</code>(普通休眠)</th><th><code>STATE_DOZE_SUSPEND</code>(深度挂起)</th></tr></thead><tbody><tr><td><code>setExact()</code> / <code>setWindow()</code></td><td>❌ 延迟至维护窗口</td><td>❌ 完全冻结</td></tr><tr><td><code>setAndAllowWhileIdle()</code></td><td>✅ 可触发(精度低)</td><td>❌ 无效</td></tr><tr><td><code>setExactAndAllowWhileIdle()</code></td><td>✅ 可触发(秒级误差)</td><td>❌ 无效</td></tr><tr><td><code>setAlarmClock()</code></td><td>✅ 立即触发</td><td>✅ 强制唤醒(需用户可见通知)</td></tr></tbody></table>
<blockquote><p><strong>结论</strong>:<br />在 <code>STATE_DOZE_SUSPEND</code> 下,<code>setExactAndAllowWhileIdle()</code><strong> 无法触发</strong>。若需极端省电模式下的可靠性,应优先使用 <code>setAlarmClock()</code> 或引导用户设置白名单)。</p></blockquote>
<p class="maodian"></p><h2>8. 更多内容</h2>
<p>有关WorkManager基础的概念和使用,详见 : Android WorkManager的概念和使用</p>
<p>到此这篇关于Android Doze低电耗休眠模式 与 WorkManager详解的文章就介绍到这了,更多相关Android Doze低电耗休眠模式内容请搜索琼殿技术社区以前的文章或继续浏览下面的相关文章希望大家以后多多支持琼殿技术社区!</p>
<div class="art_xg">
<b>您可能感兴趣的文章:</b><ul><li>Android Doze模式下Alarm定时任务实现流程详解</li><li>Android Doze模式启用和恢复详解</li><li>Android WorkManager的概念和使用详细指南</li><li>Android使用WorkManager实现缓存清理的方案</li><li>android kotlin集成WorkManager实现定时获取数据的步骤</li><li>Android WorkManager使用以及源码分析</li><li>Android WorkManager实现后台定时任务流程详解</li></ul>
</div>
</div>
<!--endmain-->
頁:
[1]