安卓性能调优:内存使用分析和方法调用优化

安卓性能调优:内存使用分析和方法调用优化

Android 应用的性能分析,优化,需要检查分析内存使用情况和方法调用情况。本文给出进行这两方面分析的工具和方法。

内存使用分析


1. 分析内存使用

虽然Android系统的Dalvik虚拟机有垃圾回收机制,但因手机内存使用存在不同于普通PC的更大的限制,内存使用方面的问题,我们更应多加注意。

  1. 一些内存使用问题会非常明显,比如内存耗尽(不足)时触发的OutOfMemoryError可能会使App直接崩溃。

  2. 另有一些内存问题则表现得不那么明显,但他们会让你的App以及系统变得越来越慢。

当有以上两种情况之一时,就得看看内存的使用情况了,是否存在:

  • 过大的对象,占用内存
  • 有些对象一直被创建,从未被释放。
2. 工具

在Android的ADT中,提供了两种工具可以用来分析内存使用

  • 对象分配相关:DDMS中的Allocation Tracker。借助这个工具可以查看对象的生成和分配情况, 可了解到对象在何时被创建,但无法了解整个App的对象分配情况。

  • Heap使用情况相关:

    1. DDMS中的Heap工具。
    2. hprof导出工具,在DDMS中导出hprof文件,在Memory Analyzer中查看。

hprof文件是Java 虚拟机的Heap快照

3. 查看Heap实时情况

安卓性能调优:内存使用分析和方法调用优化

  1. 打开DDMS,选中应用,点击Update Heap按钮
  2. 右侧Heap标签页,显示了Heap使用情况
  3. 操作应用,看哪些操作将导致内存用量增大
4. Memeory Analyzer分析内存使用情况

根据实时的Heap使用情情况,我们可以大致判断哪些操作,哪些页面可能存在内存是一共问题,但是具体的问题的需要更进一步的数据。

Allocation Tracker提供了对象分配和被引用的详细的信息

另外,还提供了一个报告,为我们分析提供参考

请在此处下载:Memeory Analyzer

我们可以通过DDMS导出hprof文件,在Memeory Analyzer中分析, 如下:

安卓性能调优:内存使用分析和方法调用优化

  1. 打开DDMS, 选中应用,点击Dump HPROF file, 等待一段时间, 10几秒甚至更长,保存hprof文件。
  2. 导出的文件为Dalvik虚拟机格式的,需要转成J2SE虚拟机格式的,否则Memeory Analyzer无法打开

    在windows中,cmd:

    cd /d D:\android\adt\adk\tools
    hprof-conv.exe D:\tmp\com.srain.cube.sample.hprof D:\tmp\com.srain.cube.sample-conv.hprof
    
  3. 在Memeory Analyzer 中打开文件 打开文件分析的过程中,会提示是否生成分析报告,分析报告会指出哪些对象是可疑的占用内存的对象。

界面展示大致如下: 安卓性能调优:内存使用分析和方法调用优化

点击Histogram:

安卓性能调优:内存使用分析和方法调用优化

各对象在列表中,可排序:

  • Shallow Heap: 占用的真正的内存大小
  • Retained Heap: 对象自身的大小 + 所维护的引用的大小

选中某个对象,List Objects -> with incoming reference / with outcoming reference 可查看引用和被应用的情况。 根据这些,加上搜索,可判断未释放的或者过大的有问题的对象的位置。

Memeory Analyzer功能强大,更多用法,点击这里


方法调用分析

App不流畅卡顿,和方法执行速度有更直接的关系。 主UI线程上的耗时操作,超过5s,系统就会提示用户,是否终止程序。 在ListView中的getView()方法,一个耗时10ms的操作就足够把你的列表卡顿得惨不忍睹。

Android框架Debug类提供了方法,记录方法调用的执行数据到一个trace文件,在代码中:

// 开始 trace文件位置: /sdcard/cube.trace
Debug.startMethodTracing("cube");

// ...
// 其他的代码

// 停止
Debug.stopMethodTracing();

在模拟器或者没SDK的真机上调试时,直接使用 /sdcard下的路径可能会有Permission deny错误,改用机身内部存储试试。

生成的trace文件,通过adb pull存到本地。

adb pull /sdcard/cube.trace D:\tmp\cube.trace

直接在ADT的eclipse中打开:

安卓性能调优:内存使用分析和方法调用优化

上图中:

  1. 上部区域(Timeline Panel)为各线程的时间线上的概况
    • 在区域1,鼠标为左右箭头状,在放大之后,可拖动缩小尺寸
    • 鼠标放在各线程时间轴区域,比如主线程的2区域,鼠标成十字状,左右拖动可以选择关注区域,选择合适的关注区域,松开鼠标,区域将放大。
  2. 下部区域(Profile Panel)为方法调用情况, 几个参数介绍如下:
    • cpu time,方法执行的真正的时间
    • real time, cpu time + 其他时间(IO wait, Thread wait)
    • Inc xxx Time, Inc 为 inclusive 缩写,本方法调用时间以及本方法内部所调用的方法(子方法)的总和
    • Excl xxx Time, Excl 为exclusive 的缩写,指的除去子方法,该方法本身执行时间
    • Calls + RecurCalls/Total, 显示父子方法调用次数占比
  3. 选中一个方法,在时间线图中会有突出显示。放大时间线图,可直观看出方法执行时间长度;看调用图,可找出该方法被调用的层次关系。
  4. 查看Excl Time 和Inc Time,分析调用关系,可找出真正耗时的方法,找出性能瓶颈。

安卓性能调优:内存使用分析和方法调用优化

上图中,Excl Time 排名第二的方法 bytesToHexString 很可能是有性能问题的。

 

转载于:https://www.cnblogs.com/liaohuqiu/p/3528876.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/109925.html原文链接:https://javaforall.net

(0)
上一篇 2021年8月27日 下午8:00
下一篇 2021年8月27日 下午8:00


相关推荐

  • 带通滤波器和带阻滤波器详细解析:(定义,区别,工作原理,经典电路图,应用)「建议收藏」

    带通滤波器和带阻滤波器详细解析:(定义,区别,工作原理,经典电路图,应用)「建议收藏」带通滤波器和带阻滤波器详细解析:(定义,区别,工作原理,经典电路图,应用)[导读]带通滤波器:它允许一定频段的信号通过,抑制低于或高于该频段的信号、干扰和噪声;带阻滤波器:它抑制一定频段内的信号,允许该频段以外的信号通过。关键词:滤波器 带通滤波器和带阻滤波器的定义  按所处理的信号分为模拟滤波器和数字滤波器两种。  按所通过信号的频段分为低通、高通、带通和带阻滤波器四种。  带通滤波器:它允许一定频段的信号通过,抑制低于或高于该频段的信号、干扰和噪声;  带阻滤波器…

    2022年5月12日
    83
  • javaclasscastexception Scala_java unchecked cast object to T

    javaclasscastexception Scala_java unchecked cast object to T在处理JSON时将一个JSONArray强转成List,在线上环境运行正常,但是换了一个环境就出现ClassCastException这个异常。编译时这个强转不会报错,但是运行时却可能出现异常。所以在对对象进行强制转换的时候一定要加以小心,想好实际的对象类型是什么,可不可以强转。

    2025年8月28日
    7
  • 用Android Studio做一个超好玩的拼图游戏,附送超详细注释的源码

    用Android Studio做一个超好玩的拼图游戏,附送超详细注释的源码在这篇博客里面 我们就来开发一款简单的拼图游戏 这款拼图游戏就和我们小时候玩的游戏是一样的 这里面的涉及到的算法不多 可以很容易学会 这次的拼图游戏项目是一个非常好的 Android 实现案例 涉及到很多常用的控件和知识点 希望大家拿到源码后 能对照着教程和注释好好学习掌握

    2026年3月16日
    2
  • 运行时异常和一般异常的区别[通俗易懂]

    运行时异常和一般异常的区别[通俗易懂]Java提供了两类主要的异常:runtimeexception和checkedexception。checked异常也就是我们经常遇到的IO异常,以及SQL异常都是这种异常。对于这种异常,JAVA编译器强制要求我们必需对出现的这些异常进行catch。所以,面对这种异常不管我们是否愿意,只能自己去写一大堆catch块去处理可能的异常。  但是另外一种异常:runtimeexc

    2026年4月17日
    6
  • C里使用CopyMemory

    C里使用CopyMemorySocket 接收到的 byte 要转换成自定义的 struct 自定义 Struct 转换成 byte 都相当麻烦用循环去转换太浪费时间了 于是想到用 CopyMemory Google 一圈终于搞定下面的代码是在 SnippetCompi 里编译通过的 C 代码 region nbsp Imports nbsp nbsp using nbsp System nbsp nbsp using nbsp S

    2026年3月19日
    2
  • oracle amm改成asmm,记AMM转变为ASMM的记录「建议收藏」

    oracle amm改成asmm,记AMM转变为ASMM的记录「建议收藏」1.修改AMM变成ASMM关闭memory参数,开启sga和pga[oracle@oradb~]$sqlplus”/assysdba”SQL*Plus:Release11.2.0.4.0Productionon星期五1月2214:07:322021Copyright(c)1982,2013,Oracle.Allrightsreserved.连接到:Orac…

    2022年6月7日
    36

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号