原文见: 在路上的blog

Android后台耗电分析及优化

  • 一、什么是耗电优化?
  • 二、耗电优化第一个方向:优化后台耗电
    • 1、唤醒锁定操作卡住(前台&后台)
    • 2、唤醒次数过多
    • 3、WLAN扫描次数过多(后台)
    • 4、后台移动网络使用量过高
  • 三、耗电优化第二个方向:让系统认为是正常耗电
    • (1)海外应用
    • (2)国内应用之华为后台资源红线标准
    • (3)经验性总结规则
  • 四、耗电监控
    • 1、google vitals不适合
    • 2、合适的耗电监控方式
      • (1)解析bugreport
      • (2)Java Hook
      • (3)插桩
  • 五、名次解释

主要参考资料:

  • Android程序性能优化之耗电优化
  • Android vitals管理中心文档

开源后台耗电分析工具: battery_alalyze

一、什么是耗电优化?

在实践中,如果我们的应用需要播放视频、获取GPS信息、需要拍照,这些耗电看起来是无法避免的。
如果发现某个应用没怎么使用(前台时间很少),但是耗电却非常多。这种情况会跟用户的预期差别很大,这种情况就需要优化。

二、耗电优化第一个方向:优化后台耗电

根据Android Vitals定义,影响后台耗电的动作如下:

  • 唤醒锁定操作卡住
  • 唤醒锁定操作卡住(后台)
  • 唤醒次数过多
  • WLAN 扫描次数过多(后台)
  • 网络使用量过高(后台)

1、唤醒锁定操作卡住(前台&后台)

应用会通过调用带有 PARTIAL_WAKE_LOCK 标记的 acquire() 来获取部分唤醒锁定。当您的应用在后台运行时,如果部分唤醒锁定保持了较长时间,则会变为卡住状态(用户看不到应用的任何部分)。 它会阻止设备进入低功耗状态。部分唤醒锁定仅应在必要时使用,并且在不再需要时立即释放。

Android Vitals 报告部分唤醒锁定卡住的条件是在以下任一时段内至少发生了一次时长达 1 小时的部分唤醒锁定:
(1)所有情况下至少 0.70% 的电池工作时段

(2)仅在后台运行时至少 0.10% 的电池工作时段

唤醒锁定操作卡住的问题发现和修复建议

2、唤醒次数过多

唤醒是 AlarmManagerAPI 中的一种机制,可让开发者设置闹钟以在指定时间唤醒设备。为设置唤醒闹钟,您的应用会调用 AlarmManager 中某个带有 RTC_WAKEUP 或 ELAPSED_REALTIME_WAKEUP 标记的 set() 方法。当唤醒闹钟触发时,设备会在执行闹钟的 onReceive() 或 onAlarm() 方法期间退出低功耗模式并保持部分唤醒锁定。如果唤醒闹钟触发次数过多,则可能会耗尽设备的电池电量。

唤醒次数过多标准:用户遇到每小时 10 次以上唤醒的电池工作时段数百分比。

  • Vital 详细信息:

    • 受影响的工作时段数:用户遇到每小时 10 次以上唤醒的电池工作时段数百分比。电池会话是指设备在两次充满电之间的间隔时间。Google 仅会在设备未充电时收集这项数据。
    • 会话数:系统已记录的会话的大概数量。
    • 第 90/99 个百分位:10%/1% 的每日工作时段中用户每小时遇到唤醒次数高于显示的值。
      最低 25%:如果您的应用发生问题的工作时段比例等于或高于显示的阈值,则系统会将此应用归在这项指标的最低 25% 区间(依据为 Google Play 上前 1000 个热门应用,按安装量统计)。

唤醒过多修复及建议

3、WLAN扫描次数过多(后台)

当应用在后台执行 WLAN 扫描时,它会唤醒 CPU,从而加快耗电速度。扫描次数过多时,设备的电池续航时间可能会明显缩短。如果某个应用处于 PROCESS_STATE_BACKGROUND 或 PROCESS_STATE_CACHED 状态,则会被视为在后台运行。

WLAN 扫描次数过多的标准:在后台运行时,应用在 0.10% 的电池工作时段内每小时执行的扫描超过 4 次。

建议:如果可能,您的应用执行 WLAN 扫描时应该是在前台运行。前台服务会自动显示通知;在前台执行 WLAN 扫描,从而让用户知道设备上发生 WLAN 扫描的原因和时间。

扫描次数过多优化:如果您的应用无法避免在后台运行期间执行 WLAN 扫描,则可能适合采用偷懒至上策略。“偷懒至上”包含三种可用于消减 WLAN 扫描次数的方法:“减少”、“推迟”和“合并”。如需了解这些方法,请参阅针对电池续航时间进行优化。

4、后台移动网络使用量过高

当应用在后台连接移动网络时,应用会唤醒 CPU 并开启无线装置。如果反复执行此操作,可能会耗尽设备的电池电量。如果某个应用处于 PROCESS_STATE_BACKGROUND 或 PROCESS_STATE_CACHED 状态,则会被视为在后台运行。

后台网络使用量过高的标准:在后台运行时,应用在 0.10% 的电池工作时段内每小时发送和接收的数据合计达 50 MB。

建议:可以将应用的移动网络使用量移至前台,提醒用户目前正在进行下载,并为他们提供暂停或停止下载的控件。为此,请调用 DownloadManager 并根据情况设置 setNotificationVisibility(int)。

三、耗电优化第二个方向:让系统认为是正常耗电

如何让系统认为是正常耗电呢?当耗电指标低于规则时,系统也就认为是正常耗电了。

(1)海外应用

海外应用主要参考Google Vitals的规则。
对于Google Vitals的后台耗电过多统计规则中的电池工作时段百分比,对于质量评估来看,较难把握。所以主要关注规则的具体指标,即相对更严格的质量要求:

(2)国内应用之华为后台资源红线标准

(3)经验性总结规则

对于国内应用来说,目前还没有非常通用且权威的后台耗电规则,根据经验,我们将监控的内容抽象成规则。
当然不同应用监控的事项或者参数都不太一样。由于每个应用的具体情况都不太一样。
下面是一些可以用来参考的简单规则。

四、耗电监控

那我们的耗电监控系统应该监控哪些内容,怎么样才能比 Android Vitals 做得更好呢?

  • 监控信息:简单来说系统关心什么,我们就监控什么,而且应该以后台耗电监控为主。类似 Alarm wakeup、WakeLock、WiFi scans、Network 都是必须的,其他的可以根据应用的实际情况。如果是地图应用,后台获取 GPS 是被允许的;如果是计步器应用,后台获取 Sensor 也没有太大问题。
  • 现场信息:监控系统希望可以获得完整的堆栈信息,比如哪一行代码发起了 WiFi scans、哪一行代码申请了 WakeLock 等。还有当时手机是否在充电、手机的电量水平、应用前台和后台时间、CPU 状态等一些信息也可以帮助我们排查某些问题。

1、google vitals不适合

缺点:

  • 耗电规则无法修改
  • 无法拿到堆栈和其他电池信息
  • 国内应用无法使用

2、合适的耗电监控方式

(1)解析bugreport

通常大家可能会使用Battery Historian来分析后台耗电,但是不够灵活。比如需要人工查看各资源使用情况及是否达标。所以用python实现了一个简单的分析bugreport文件的小工具;
核心代码是刚做测开半年左右写的,比较乱且水平有限,大家轻拍,也欢迎大家参与优化。

  • 实现逻辑:

    • 重置电池统计信息和历史记录(dumpsys batterystats --reset)
    • 打开详细的wakelock数据开关,日志量较大,一般可正常保存3个小时以内。
      • dumpsys batterystats --enable full-wake-history --启用
      • dumpsys batterystats --disable full-wake-history --关闭
    • 导出bugreport文件
      • Android 7.0 and higher: adb bugreport > bugreport.zip
      • Android 6.0 and lower: adb bugreport > bugreport.txt
    • 利用battery_analyze生成后台耗电报告

(2)Java Hook

Hook 方案的好处在于使用者接入非常简单,不需要去修改自己的代码。下面我以几个比较常用的规则为例,看看如果使用 Java Hook 达到监控的目的。

  • WakeLock:WakeLock 用来阻止 CPU、屏幕甚至是键盘的休眠。类似 Alarm、JobService 也会申请 WakeLock 来完成后台 CPU 操作。WakeLock 的核心控制代码都在PowerManagerService中,实现的方法非常简单。

// 代理 PowerManagerService
ProxyHook().proxyHook(context.getSystemService(Context.POWER_SERVICE), "mService", this);@Override
public void beforeInvoke(Method method, Object[] args) {// 申请 Wakelockif (method.getName().equals("acquireWakeLock")) {if (isAppBackground()) {// 应用后台逻辑,获取应用堆栈等等     } else {// 应用前台逻辑,获取应用堆栈等等}// 释放 Wakelock} else if (method.getName().equals("releaseWakeLock")) {// 释放的逻辑    }
}
  • Alarm:Alarm 用来做一些定时的重复任务,它一共有四个类型,其中ELAPSED_REALTIME_WAKEUP和RTC_WAKEUP类型都会唤醒设备。同样,Alarm 的核心控制逻辑都在AlarmManagerService中,实现如下:
// 代理 AlarmManagerService
new ProxyHook().proxyHook(context.getSystemService
(Context.ALARM_SERVICE), "mService", this);public void beforeInvoke(Method method, Object[] args) {// 设置 Alarmif (method.getName().equals("set")) {// 不同版本参数类型的适配,获取应用堆栈等等// 清除 Alarm} else if (method.getName().equals("remove")) {// 清除的逻辑}
}
  • 其他:对于后台 CPU,我们可以使用卡顿监控相关的方法。对于后台网络,同样我们可以通过网络监控相关的方法。对于 GPS 监控,我们可以通过 Hook 代理LOCATION_SERVICE。对于 Sensor,我们通过 Hook SENSOR_SERVICE中的“mSensorListeners”,可以拿到部分信息。

通过 Hook,我们可以在申请资源的时候将堆栈信息保存起来。当我们触发某个规则上报问题的时候,可以将收集到的堆栈信息、电池是否充电、CPU 信息、应用前后台时间等辅助信息也一起带上。

(3)插桩

虽然使用 Hook 非常简单,但是某些规则可能不太容易找到合适的 Hook 点。而且在 Android P 之后,很多的 Hook 点都不支持了。
出于兼容性考虑,我首先想到的是写一个基础类,然后在统一的调用接口中增加监控逻辑。以 WakeLock 为例:

public class WakelockMetrics {// Wakelock 申请public void acquire(PowerManager.WakeLock wakelock) {wakeLock.acquire();// 在这里增加 Wakelock 申请监控逻辑}// Wakelock 释放public void release(PowerManager.WakeLock wakelock, int flags) {wakelock.release();// 在这里增加 Wakelock 释放监控逻辑}
}

Facebook 也有一个耗电监控的开源库Battery-Metrics,它监控的数据非常全,包括 Alarm、WakeLock、Camera、CPU、Network 等,而且也有收集电量充电状态、电量水平等信息。
Battery-Metrics 只是提供了一系列的基础类,在实际使用中,接入者可能需要修改大量的源码。但对于一些第三方 SDK 或者后续增加的代码,我们可能就不太能保证可以监控到了。这些场景也就无法监控了,所以 Facebook 内部是使用插桩来动态替换。
遗憾的是,Facebook 并没有开源它们内部的插桩具体实现方案。大家可以自行搜索不同插桩方案的实现。
插桩方案使用起来兼容性非常好,并且使用者也没有太大的接入成本。但是它并不是完美无缺的,对于系统的代码插桩方案是无法替换的,例如 JobService 申请 PARTIAL_WAKE_LOCK 的场景。

五、名次解释

  • 电池工作时段:是指两次电池充满电的时间间隔。

Android后台耗电分析及优化相关推荐

  1. Android应用耗电分析与优化

    应用的耗电是个长期持续优化的事情,而且随着Android系统的不断更新,系统本身也提供了越来越详细的信息来辅助统计分析耗电.这里基于Android 6.0介绍一些耗电分析与优化的思路及实践,一方面可以 ...

  2. Android 系统性能优化(30)---Android性能全面分析与优化方案研究

    Android 性能优化 1.结合以下四个部分讲解: 性能问题分类 性能优化原则和方法 借助性能优化工具分析解决问题 性能优化指标 2性能问题分类 1.渲染问题:过度绘制.布局冗杂 2.内存问题:内存 ...

  3. 耗电优化(上):Android App 耗电分析

    这里写目录标题 1. 电量和硬件 1.1 App 通过使用硬件模块消耗相应的电能 1.2 资源调度机制是厂商功耗优化最重要的手段 2. 电量和应用程序 2.1 评估不同应用程序的耗电情况 结论:把电量 ...

  4. android app耗电分析方法

    这是一篇讲述应用耗电的文章,围绕 Android 电量采集机制及第二代 Battery Historian 分析工具讲述.文从数据采集.导出.环境搭建.解读报告的角度出发,从细节讲解整个流程.和大谈概 ...

  5. 几乎是史上最全最实用的Android性能全面分析与优化方案研究

    结合以下四个部分讲解: 性能问题分类 性能优化原则和方法 借助性能优化工具分析解决问题 性能优化指标 性能问题分类 1.渲染问题: 过度绘制.布局冗杂 2.内存问题: 内存浪费(内存管理).内存泄漏 ...

  6. Android App耗电分析

    项目中用到了一个比较坑爹的问题,就是App的耗电问题,发现自己开发的时候没有注意这个问题,一发布给用户就暴露了这个我平常不注意的问题挺蛋疼,平常开发我都觉得耗电是很正常的,再耗电也在可接受范围内,但是 ...

  7. BatteryHistorian Android手机耗电分析神器

    Battery-Historian是谷歌推出的一款专门分析Bugreport的工具,是谷歌2015年I/O大会上推出的一款检测运行在android5.0(Lollipop)及以后版本的设备上电池的相关 ...

  8. android GMS 耗电分析

    安装GMS app 的system耗电比较严重, 尤其是Checkin Service , 在网络无法访问(目前来说无法翻墙就访问不了)的情况下会长期持锁,从而导致手机无法睡眠, 耗电比较严重 作者: ...

  9. Android APP性能分析方法及工具

    近期读到<Speed up your app>一文.这是一篇关于Android APP性能分析.优化的文章.在这篇文章中,作者介绍他的APP分析优化规则.使用的工具和方法.我觉得值得大家借 ...

最新文章

  1. C++ [](){} 匿名函数 lambda表达式
  2. Android11还能自定义相机吗,安卓用户又少了一项自由,Android 11不再支持更改默认相机程序...
  3. 【视频课】模型部署课程更新!ncnn框架快速实践!
  4. python 类继承与子类实例初始化
  5. [汇编语言]-第十章 ret,retf,call指令
  6. jquery点击非div区域隐藏div
  7. oracle 取整点的数据,Oracle SQL语句操作数字:取整、四舍五入及格式化
  8. Oracle归档日志与非归档日志的切换及路径设置
  9. Maven配置之pom.xml(一)
  10. Akka构建Reactive应用《one》
  11. stm32气压传感器 带探头的_几种常用传感器
  12. redhat 安装 snort
  13. cpython安装_Cython安装没有找到Python.h文件?
  14. 2017 CCPC Final小结 By JSB @ Reconquista
  15. Zynq7000硬件开发之电源供电系统(PDN)设计(一)
  16. 手机pdf文件转语音_录音语音转文字手机版下载-录音语音转文字免费版下载v1.0.0...
  17. stm8s103k3 周期 捕获_STM8S103之tim1捕获周期及占空比
  18. duilib 添加自定义list一例
  19. 微信网页开发(2)--使用微信开发者工具
  20. 【教程】:Photoshop ps 图片批处理

热门文章

  1. linux下面用ssh替代rsh
  2. Spark基础(五)SparkSteaming
  3. 关于OPCUA的配套规范
  4. CSGO 控制台 准星详细设置
  5. 使用 Scrapy 框架对重复的 url 无法获取数据,dont_filter=True
  6. 【译】如何学习机器学习
  7. select、poll、epoll之间的区别总结摘自http://www.cnblogs.com/Anker/p/3265058.html
  8. android打开微信运动,小米手机怎么开启微信运动?
  9. 常见健身器材EN ISO 20957认证标准有哪些
  10. php连接mongodb数据库报错No suitable servers found