一.电池省电优化(Doze和App Standby模式)
从Android6.0开始,系统提供了两种省电功能(延长电池寿命和使用时间):Doze和App Standby
Doze和App Standby模式会延缓CPU和网络活动实现节能;
1.Doze模式
1.进入Doze模式
在手机未充电,完全静止且熄屏一段时间后,Android系统会自动进入Doze模式
2.Doze模式效果
1.网络访问被挂起
2.Wake Locks被无视
3.AlarmManager被推迟到下一个maintenance window窗口,
除非使用AlarmManager新方法: setAndAllowWhileIdle(),setExactAndAllowWhileIdle(),setAlarmClock()
4.WiFi扫描被停止
5.SyncAdapter同步工作被停止
6.JobScheduler定时任务被停止
Doze模式的五种状态
ACTIVE: 活动状态
INACTIVE: 屏幕关闭进入非活动状态
IDLE_PENDING: 每隔30分钟让App进入预备状态
IDLE: 空闲状态
IDLE_MAINTENANCE:处理挂起任务
系统进入Doze模式后,每隔一段时间(30分钟)会有一小段时间(30s)供APP处理被挂起任务,
但随着时间推移,间隔时间会变长,以此减少电量消耗
3.退出Doze模式
1.手机充电
2.手机移动
3.手机屏幕打开
2.App Standby模式
1.进入App Standby模式
长时间未被用户使用的App,将进入App Standby状态(被标志为空闲状态)
2.退出App Standby模式
1.用户主动启动该App
2.该App有前台进程(前台activity/前台service),被其它前台进程启动
3.该App在锁屏或通知栏有可见的Notification
4.Android设备充电时,会将所有Standby状态的App释放,处理挂起任务
5.如果App空闲时间很长,系统将允许App一天一次访问网络
3.App Standby和Doze区别
App Standby不需要屏幕关闭,App进入后台一段时间,网络也会受到限制
Doze模式需要屏幕关闭(通常晚上睡觉或长时间屏幕关闭才会进入)
3.阻止电池优化(白名单)
Android6.0及更高版本提供电池优化白名单,App加入白名单可逃脱Doze和App Standby限制,
处于白名单中的App也会受到一定限制: Jobs和Syncs以及常规Alarms也会被推迟;
用户手动设置App进入白名单: 设置>电池>电池优化白名单
App检测是否在白名单: PowerManager.isIgnoringBatteryOptimizations()
App请求加入白名单:
1.通过创建Intent ACTION_IGNORE_BATTERY_OPTIMIZATION_SETTINGS 引导用户前往电池优化界面,加入白名单
2.持有权限 REQUEST_IGNORE_BATTERY_OPTIMIZATIONS 可直接触发系统Dialog来提醒用户,加入白名单(无需进入设置界面)
3.创建Intent ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS 触发上述系统Dialog
二.AlarmManager定时闹钟失效
1.App被Kill后,AlarmManager失效(只有App进程在运行时,才会收到系统定时AlarmManager通知)
添加守护进程,相互监听重启; 提醒用户加入锁屏清理白名单
2.手机重启,AlarmManager任务失效
监听重启广播,重新设置定时闹钟
3.从Android4.4(API19)开始,AlarmManager机制修改,set(),setRepeating()定时不再精确,甚至setRepeating只生效一次(不会重复)
新增精确定时方法: setExact(),setWindow(),setAlarmClock(),
4.从Android6.0(API23)开始,进入Doze模式(省电优化),AlarmManager被延缓
新增精确定时方法: setAndAllowWhileIdle(),setExactAndAllowWhileIdle()
从Android4.4(API19)开始,新增精准定时方法都是一次性闹钟,没有重复定时的方法,
所以当需要重复周期闹钟,只能在下一次唤醒时重新设置定时,间接实现重复闹钟!
精确定时一次Demo如下:
void setWakeAtTime(Context cxt, int delay) {
PendingIntent pi = PendingIntent.getService(cxt,0,new Intent(cxt, xxService.class),PendingIntent.FLAG_UPDATE_CURRENT);
AlarmManager alarm = (AlarmManager) cxt.getSystemService(Context.ALARM_SERVICE);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) //Android 6,针对省电优化
alarm.setExactAndAllowWhileIdle(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime() + delay, pi);
else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) //Android 4.4,针对set不准确
alarm.setExact(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime() + delay, pi);
else
alarm.set(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime() + delay, pi);
}
简书: http://www.jianshu.com/p/585ca251b871
CSDN博客: http://blog.csdn.net/qq_32115439/article/details/78848386
GitHub博客: http://lioil.win/2017/12/19/Android-Alarm_Doze_Standby.html
Coding博客: http://c.lioil.win/2017/12/19/Android-Alarm_Doze_Standby.html