使用DevEco Studio Profiler工具的ArkTS内存泄漏分析案例

描述

在鸿蒙 ArkTS 与 Native(C/C++)交互的开发场景中,ArkTS 对象被 Native 引用导致的内存泄漏问题是开发者的一大痛点。

这类问题的传统定位方式高度依赖开发者具备深厚的技术积累,不仅要求开发者要精通 ArkTS 引擎垃圾回收机制、鸿蒙 NAPI 交互模型等基础理论,还要深入理解应用代码的实现细节,尤其在应用由多个模块组成、模块间依赖关系复杂的场景下,问题定位复杂度倍增。

这种高门槛导致许多开发者,尤其是初学者,在面对 ArkTS 对象被Native引用导致内存泄漏问题时往往无从下手,只能通过参考典型案例、分析大量业务代码来进行问题定位,效率较为低下。

为了帮助开发者更高效地定位和解决此类内存泄漏问题,DevEco Studio 的 Profiler 工具迎来了重大升级!本次更新增强了 Allocation 内存分析能力,新增了 ArkTS Snapshot 泳道,支持 Local Handle 和 Global Handle 采样模式,在跨语言调用场景中,打通 ArkTS 和 C++ 语言边界,内存泄漏大小和泄漏对象信息易观测,问题定位效率倍增。

本文中,我们将结合 DevEco Studio Profiler 工具,系统梳理 ArkTS 对象被Native 引用导致内存泄漏的定位思路、实操步骤与最佳实践。

为什么 ArkTS 对象会被 Native “锁死”?

ArkTS 对象被 Native 引用导致内存泄漏的根因在于 Native 代码通过napi_ref 或不当使用 napi_value ,导致长期持有 ArkTS 对象,形成了“隐形强引用”,阻断了 ArkTS 引擎的垃圾回收。

在鸿蒙 NAPI 交互模型中,Native 代码引用 ArkTS 对象主要依赖两种句柄:Local Handle (napi_value): 通常指在 Native 代码执行上下文中创建的、作用域较短的引用,由 handle scope 来管理。

Global Handle (napi_ref): 这是一种由开发者自行管理生命周期的强引用。

ArkTS 引擎的垃圾回收是基于“可达性分析”算法实现的,当 ArkTS 侧所有引用(变量、闭包等)被清除后,对象理论上应被回收。一旦 Native 代码通过 napi_ref 或 napi_value不当持有该对象,这两种句柄会在 GC 图中成为一个“根节点”。 即使 ArkTS 侧已彻底断开引用,GC 仍会判定该对象“可达”,导致无法回收。久而久之,未释放的句柄堆积,演变为内存溢出问题。

升级后的Allocation 内存分析工具能做什么?

在定位 ArkTS 对象被 Native 引用引发的内存泄露问题时,开发者常受限于工具或日志信息不足,难以追溯具体是哪块 Native 代码在持有着它。这导致泄漏边界模糊、引用链断裂,定界分析极为困难。升级后的 Allocation 工具实现了关键突破——支持 ArkTS 对象和 Native 引用分配调用栈 的双向关联,可精准追溯持有权属,快速还原跨语言引用关系链,大幅降低定位门槛。

ArkTS内存泄漏分析案例

案例背景

现象:本案例中,通过复现‘首页至消息页’的反复进出操作,观察到应用内存占用呈现"阶梯式持续增长"趋势。在循环操作10次后,应用出现显著卡顿现象。

初步判断:典型的“阶梯式内存增长”,高度疑似内存泄漏。

分析流程

步骤1:Memory泳道确认泄漏

1、打开 DevEco Studio,连接真机,点击 Profiler 工具 Realtime Monitor(也可使用 snapshot 模板的 Memory 录制观察,以下是使用 Realtime Monitor 观察)。

2、启动应用,选择设备与应用进程。

3、点击录制按钮,在设备上重复操作:进入消息页面 → 停留 2 秒 → 返回首页,重复多次。

4、观察内存曲线:

正常预期:每次退出后内存回落至基线附近,整体呈锯齿状。

实际现象:曲线呈阶梯状上升,多次操作后内存增长至 314.1 MB 且无明显回落。

确认存在内存泄漏,进入下一步。

内存泄漏

步骤2:堆快照对比定位异常对象

1、在 Profiler 中切换到 Snapshot 模板,请参考使用 Snapshot 模板基本操作:

Snapshot 模板基本操作:

https://developer.huawei.com/consumer/cn/doc/harmonyos-guides/ide-snapshot-basic-operations

选择 Profiler 工具 → 选择设备与应用进程 → 选择 Snapshot 模板 → 创建Session  → 启动录制

内存泄漏

2、 抓取快照 1 :首次在进入消息页前,点击“Take Heap Snapshot”。

内存泄漏

3、抓取快照 2 :重复进出消息页面 7 次后,回到首页,点击 “Take Heap Snapshot”,再抓取第二次快照,并停止录制。

内存泄漏

4、在快照对比视图 Comparison 中,选择 CompareTo Snapshot 1 。

内存泄漏

5、查看对象新增销毁情况,优先关注:

操作次数的整数倍或整数倍+1( export 出的对象本身也有一条引用链);

业务对象,即 Constructor 的结构为包名/模块名/文件路径#泄漏对象 

内存泄漏

关键发现:

对比结果中,Proxy 对象数量异常,创建 70 个未销毁。

确认 Test 对象实例泄漏。

步骤 3 :追踪引用链

1、点击  “Shortest Paths”  获得如下 Proxy 对象实例的最短引用链,发现该 Proxy 对象的 Distance 为 1 。说明其为 GC Root 直接持有的对象,被 Native 侧直接持有未释放,即 JS 对象被 Local Handle / Global Handle 引用导致内存泄漏。

内存泄漏

2、通过 Proxy 对象的 Distance 为 1 的 References ,确认内存泄漏是由 JS 对象被 Global Handle 引用导致。

内存泄漏

步骤4:分析 Local Handle / Global Handle 类型持有导致泄漏问题

基于步骤3分析得到 Proxy 对象实例可能被 Local Handle / Global Handle 引用导致内存泄露,我们可以通过以下步骤继续分析定位:

1、配置 Allocation 录制模板并捕获数据

打开 DevEco Studio:确保你的工程已加载,并连接了目标设备或模拟器。

进入 Profiler 模块:在主界面下方菜单栏,找到并点击 Profiler 选项卡。

选择应用进程:运行应用,并在 “区域 2 ”选择目标设备和应用进程。

创建 Allocation 录制模板:选择 “Allocation” 并点击 Create Session 创建录制模板

内存泄漏

配置录制参数

配置模式:选择详情模式(即关闭 Statistics Mode )。当前仅详情模式支持进行 ArkTS 和 Native 的关联分析。

配置开关:勾选“Local Handle”和“Global Handle”,这是关键配置。这将使Allocation 专门捕获与 JS-NAPI 句柄相关的内存分配事件。

如果底层镜像不支持该功能,则会提示“当前镜像版本不支持,请升级镜像”。
内存泄漏

配置泳道范围:勾选 ArkTS Snapshot 泳道。这将使 Allocation 在录制结尾时自动抓取一份 Snapshot 快照用于关联分析。

内存泄漏

启动录制:勾选了“Local Handle”开关后,如果是在应用本生命周期内首次录制local handle 数据,会触发弹窗请求重启应用以便录制对应信息,此时点击 OK 允许重启即可。

内存泄漏

运行应用程序:运行目标应用,执行相关被怀疑引入内存泄露的业务操作,持续一段时间以增加内存压力和捕获更多数据。

停止录制:自动触发抓取一份 Snapshot 快照用于关联分析。点击快照,查找到疑似泄漏对象 Proxy 。 

内存泄漏

2、泄漏对象关联分析

定位可疑 ArkTS 对象:选中一个怀疑被泄漏的 ArkTS 对象实例(或对象类型),查看扩展标签页。  内存泄漏

查看 Native List :若某个 ArkTS 对象的 distance 值为 1,则可以通过扩展标签页中的 Native List 标签页,查看所有当前与该 JS 对象关联的 Native 句柄引用,以确认该 JS 对象是被 Local Handle 或 Global Handle 引用的对象。

内存泄漏

关键信息:

(1)句柄类型:调用栈底层的符号 ArkGlobalHandle 或 ArkLocalHandle 判断泄漏类型

(2)调用栈:通过调用栈,可以定位到应用的 Native 代码(可能是 ArkUI 框架代码或你自己代码)中创建 napi_ref 的地方。

(3)注意点:

如果该 JS 对象节点不是一个被 Local Handle 或者 Global Handle 引用的对象,则会提示 "No Detail";

如果该 JS 对象确实是一个被 Local Handle 或者 Global Handle 引用的对象,但是对应的 native 内存的申请事件已经在此次录制之前完成内存分配,本次录制结果则无法展示对应的内存申请调用栈,需要重新录制,录制时需要注意将录制时执行的业务逻辑范围调整的尽量更早一些。

3、分析内存分配调用栈

排查调用栈:“Native List”标签页中的调用栈,找到对应业务代码

关键排查点: 

检查是否在适当的时候调用了对应的句柄释放接口如 napi_delete_reference等。  

梳理这段 Native 代码需要引用 ArkTS 对象的合理性,识别这个引用的生命周期是否过长,是否应该在某个条件满足后被释放;

对于 napi_ref ,其引用计数是关键。确保在不再需要引用时正确调用了 napi_delete_reference 。注意,引用计数可能因其他代码路径的创建或删除操作而意外增减。

检查句柄的作用域是否合理。local handle 是否应该只在 JS 执行上下文切换前使用?global handle 是否真的需要在整个应用生命周期都有效  内存泄漏

修复验证

1、重新运行应用,再次使用 Memory 泳道监控。

2、重复多次进出消息中心页。

3、验证结果:

内存曲线恢复锯齿状,每次退出后回落至基线。

再次抓取快照对比,多次操作后业务对象实例数量无明显增长。

泄漏问题已修复。

内存泄漏

体验与推广

DevEco Studio Profile 工具的跨语言交互内存泄漏定位能力已就绪,欢迎开发者前往 DevEco Studio 体验最新内存分析功能,助力开发者提高内存问题定位效率。

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分