android进程内存查看 使用内存分析器查看应用的内存使用情况

内存分析器是 Android Profiler 中的一个组件,可帮助开发者识别可能会导致应用卡顿、冻结甚至崩溃的内存泄漏和内存抖动。它显示一个应用内存使用量的实时图表,让开发者可以捕获堆转储、强制执行垃圾回收以及跟踪内存分配。

打开内存分析器的步骤如下:

  1. 依次点击 View > Tool Windows > Profiler(也可以点击工具栏中的 Profile 图标 )。
  2. 从 Android Profiler 工具栏中选择要分析的设备和应用进程,如果您已通过 USB 连接设备但系统未列出该设备,请确保您已启用 USB 调试。
  3. 点击 MEMORY 时间轴上的任意位置以打开内存分析器
1 内存分析器概览

当首次打开内存分析器时,可以看到一条表示应用内存使用量的详细时间轴,并可使用各种工具强制执行垃圾回收、捕获堆转储以及记录内存分配。

android进程内存查看 使用内存分析器查看应用的内存使用情况(1)

如上图所示,内存分析器的默认视图包括以下各项:

  1. 用于强制执行垃圾回收事件的按钮。
  2. 用于捕获堆转储的按钮。只有在连接到搭载 Android 7.1(API 级别 25)或更低版本的设备时,系统才会在堆转储按钮右侧显示用于记录内存分配情况的按钮。
  3. 用于指定性能分析器多久捕获一次内存分配的下拉菜单。选择适当的选项可帮助开发者在进行性能剖析时提高应用性能。
  4. 用于缩放时间轴的按钮。
  5. 用于跳转到实时内存数据的按钮。
  6. 事件时间轴,显示活动状态、用户输入事件和屏幕旋转事件。
  7. 内存使用量时间轴,它会显示以下内容:

一个堆叠图表,显示每个内存类别当前使用多少内存,如左侧的 y 轴以及顶部的彩色键所示。

一条虚线,表示分配的对象数,如右侧的 y 轴所示。

每个垃圾回收事件的图标。

在 Android 8.0 及更高版本上,系统会一律为可调试的应用启用高级性能剖析,如果使用的是搭载 Android 7.1 或更低版本的设备,并非所有分析数据在默认情况下都可见。

2 内存计算方式

android进程内存查看 使用内存分析器查看应用的内存使用情况(2)

  • Java:从 Java 或 Kotlin 代码分配的对象的内存。
  • Native:从 C 或 C 代码分配的对象的内存。
  • 即使应用中不使用 C ,也可能会看到此处使用了一些原生内存,因为即使您编写的代码采用 Java 或 Kotlin 语言,Android 框架仍使用原生内存代表您处理各种任务,如处理图像资源和其他图形。
  • Graphics:图形缓冲区队列为向屏幕显示像素(包括 GL 表面、GL 纹理等等)所使用的内存。(请注意,这是与 CPU 共享的内存,不是 GPU 专用内存。)
  • Stack:应用中的原生堆栈和 Java 堆栈使用的内存。这通常与应用运行多少线程有关。
  • Code:应用用于处理代码和资源(如 dex 字节码、经过优化或编译的 dex 代码、.so 库和字体)的内存。
  • Others:应用使用的系统不确定如何分类的内存。
  • Allocated:应用分配的 Java/Kotlin 对象数。此数字没有计入 C 或 C 中分配的对象。
3 查看内存分配情况

内存分配情况图表显示内存中每个 Java 对象和 JNI 引用的分配方式:

  • 分配了哪些类型的对象以及它们使用多少空间。
  • 每个分配的堆栈轨迹,包括在哪个线程中。
  • 对象在何时被取消分配(仅当使用搭载 Android 8.0 或更高版本的设备时)。

如需录制 Java 和 Kotlin 分配情况,可以选择 Record Java / Kotlin allocations,然后选择 Record

如果设备搭载的是 Android 8 或更高版本,则内存分析器界面将转换为显示正在进行录制的单独屏幕。可以与录制上方的迷你时间轴进行交互(例如,更改选择范围)。如需完成录制,选择 Stop 图标

android进程内存查看 使用内存分析器查看应用的内存使用情况(3)

android进程内存查看 使用内存分析器查看应用的内存使用情况(4)

4 如需检查分配记录,请按以下步骤操作:

浏览列表以查找堆计数异常大且可能存在泄漏的对象。为帮助查找已知类,点击 Class Name 列标题以按字母顺序排序。然后,点击一个类名称。此时右侧将出现 Instance View 窗格,显示该类的每个实例。

android进程内存查看 使用内存分析器查看应用的内存使用情况(5)

Instance View 窗格中,点击一个实例。此时下方将出现 Call Stack 标签页,显示该实例被分配到何处以及在哪个线程中。

Call Stack 标签页中,右键点击任意行并选择 Jump to Source,以在编辑器中打开该代码。

5 捕获堆转储

堆转储显示在捕获堆转储时 对应的应用中哪些对象正在使用内存。特别是在长时间的用户会话后,堆转储会显示不应再位于内存中却仍在内存中的对象,从而识别内存泄漏,捕获堆转储后,可以查看以下信息:

  • 应用分配了哪些类型的对象,以及每种对象有多少。
  • 每个对象当前使用多少内存
  • 在代码中的什么位置保持着对每个对象的引用
  • 对象所分配到的调用堆栈

点击 Capture heap dump,然后选择 Record,在转储堆期间,Java 内存量可能会暂时增加,这是很正常的现象,因为堆转储与对应的应用发生在同一进程中,并需要一些内存以收集数据。

android进程内存查看 使用内存分析器查看应用的内存使用情况(6)

在类列表中,可以查看以下信息

  • Allocations:堆中的分配数。
  • Native Size:此对象类型使用的原生内存总量(以字节为单位)。只有在使用 Android 7.0 及更高版本时,才会看到此列。
  • 您会在此处看到采用 Java 分配的某些对象的内存,因为 Android 对某些框架类(如 Bitmap)使用原生内存。
  • Shallow Size:此对象类型使用的 Java 内存总量(以字节为单位)。
  • Retained Size:为此类的所有实例而保留的内存总大小(以字节为单位)。

从左侧的菜单中,选择需检查的堆:

  • default heap:当系统未指定堆时。
  • app heap:您的应用在其中分配内存的主堆。
  • image heap:系统启动映像,包含启动期间预加载的类。此处的分配确保绝不会移动或消失。
  • zygote heap:写时复制堆,其中的应用进程是从 Android 系统中派生的。

从右侧的菜单中,选择如何安排分配:

  • Arrange by class:根据类名称对所有分配进行分组。这是默认值。
  • Arrange by package:根据软件包名称对所有分配进行分组。
  • Arrange by callstack:将所有分配分组到其对应的调用堆栈。只有在记录分配期间捕获堆转储时,此选项才有效。即便如此,堆中也很可能有在您开始记录之前分配的对象,所以会先显示这些分配,直接按类名称列出它们。

点击一个类名称可在右侧打开 Instance View 窗口(如下图 所示)。列出的每个实例都包含以下信息:

  • Depth:从任意 GC 根到选定实例的最短跳数。
  • Native Size:原生内存中此实例的大小。 只有在使用 Android 7.0 及更高版本时,才会看到此列。
  • Shallow Size:Java 内存中此实例的大小。
  • Retained Size:此实例所支配内存的大小

android进程内存查看 使用内存分析器查看应用的内存使用情况(7)

6 内存分析器中的泄漏检测

在内存分析器中分析堆转储时,可以过滤 Android Studio 认为可能表明应用中的 Activity 和 Fragment 实例存在内存泄漏的分析数据。

过滤器显示的数据类型包括:(需要 首先需要 捕获堆转储)

  • 已销毁但仍被引用的 Activity 实例。
  • 没有有效的 FragmentManager 但仍被引用的 Fragment 实例。

在某些情况(如以下情况)下,过滤器可能会产生误报:

  • 已创建 Fragment,但尚未使用它。
  • 正在缓存 Fragment,但它不是 FragmentTransaction 的一部分。

android进程内存查看 使用内存分析器查看应用的内存使用情况(8)

,

免责声明:本文仅代表文章作者的个人观点,与本站无关。其原创性、真实性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容文字的真实性、完整性和原创性本站不作任何保证或承诺,请读者仅作参考,并自行核实相关内容。文章投诉邮箱:anhduc.ph@yahoo.com

    分享
    投诉
    首页