鸿蒙ArkTS与React Native JS双向通信实战教程

本文专注讲解鸿蒙ArkTS与React Native JS之间的双向通信原理和实战,涵盖从JS侧到ArkTS侧再到C++胶水层的完整链路。


总体架构理解

  • JS侧(React Native业务代码) :开发者日常写的JS/TS文件。
  • C++胶水层 :鸿蒙适配的桥接层,实现消息和数据的传递,配合TurboModule/JSI等机制。
  • ArkTS侧 :鸿蒙原生能力、页面、组件等实现,负责最终的业务落地。

通信流程简图如下:

scss 复制代码
JS (RN) <==> C++ (胶水层) <==> ArkTS (原生)

一、JS(RN)向ArkTS通信全流程

1.1 JS侧调用TurboModule

(1)声明TurboModule接口

typescript 复制代码
// src/specs/v2/NativeCalculator.ts
import type {TurboModule} from 'react-native/Libraries/TurboModule/RCTExport';
import {TurboModuleRegistry} from 'react-native';

export interface Spec extends TurboModule {
  add(a: number, b: number): Promise<number>;
}
export default TurboModuleRegistry.get<Spec>('RTNCalculator') as Spec | null;

(2)实际调用

javascript 复制代码
import { RTNCalculator } from 'rtn-calculator';

const result = await RTNCalculator.add(3, 7);
// result 应为 10,实际走到了原生

1.2 C++胶水代码实现TurboModule

(1)胶水头文件生成(示例)

cpp 复制代码
// generated/RTNCalculator.h
#pragma once
#include "RNOH/TurboModule.h"
namespace rnoh {
class RTNCalculator : public ArkTSTurboModule {
public:
  RTNCalculator(const Context ctx, const std::string name);
};
}

(2)实现方法与注册

cpp 复制代码
// generated/RTNCalculator.cpp
#include "RTNCalculator.h"
namespace rnoh {
RTNCalculator::RTNCalculator(const Context ctx, const std::string name): ArkTSTurboModule(ctx, name) {
  methodMap_ = {
    { "add", { 2, [](facebook::jsi::Runtime& rt, facebook::react::TurboModule& turboModule, const facebook::jsi::Value* args, size_t count) {
      // 直接转发到ArkTS
      return static_cast<ArkTSTurboModule&>(turboModule).callAsync(rt, "add", args, count);
    }}}
  };
}
}

说明: 这里等价于将参数转发到ArkTS侧,由ArkTS实现业务。

1.3 ArkTS侧实现能力

typescript 复制代码
// entry/src/main/ets/turbomodule/CalculatorModule.ts
import { TurboModule } from '@rnoh/react-native-openharmony/ts';
import { TM } from '@rnoh/react-native-openharmony/generated/ts';
export class CalculatorModule extends TurboModule implements TM.RTNCalculator.Spec {
  add(a: number, b: number): Promise<number> {
    return Promise.resolve(a + b);
  }
}

1.4 回传结果到JS侧

ArkTS实现Promise返回,C++胶水自动转成JS的Promise,JS拿到最终结果。


二、ArkTS向JS(RN)通信全流程

2.1 ArkTS侧向JS侧发事件(DeviceEventEmitter示例)

typescript 复制代码
// ArkTS侧
this.ctx.rnInstance.emitDeviceEvent("customEvent", { foo: 123 });

2.2 C++胶水代码转发事件

cpp 复制代码
// C++实现
// 对应 emitDeviceEvent 方法内部
void RNInstance::emitDeviceEvent(const std::string& eventName, const folly::dynamic& payload) {
  // 实际会通过 JSI 调用 JS 侧的 DeviceEventEmitter.emit
  // 伪代码示例:
  auto jsCallback = ...; // 得到JSEmit函数
  jsCallback(eventName, payload);
}

2.3 JS侧监听事件

javascript 复制代码
import { DeviceEventEmitter } from 'react-native';

DeviceEventEmitter.addListener('customEvent', (data) => {
  // data = { foo: 123 }
  // 这里处理ArkTS侧发来的消息
});

三、JS(RN)侧主动监听Native事件

同上,JS侧通过DeviceEventEmitter或TurboModule的回调接口监听原生主动推送过来的事件。


四、ArkTS与JS直接消息通信(消息总线机制)

ArkTS向C++侧发消息:

typescript 复制代码
// ArkTS
this.ctx.rnInstance.postMessageToCpp("SAMPLE_MESSAGE", { foo: "bar" });

C++胶水层Observer监听:

cpp 复制代码
class MyComponentInstance : public CppComponentInstance<...>, public ArkTSMessageHub::Observer {
public:
  MyComponentInstance(Context context)
    : CppComponentInstance(std::move(context)), ArkTSMessageHub::Observer(m_deps->arkTSMessageHub) {}
  void onMessageReceived(ArkTSMessage const& message) override {
    if (message.name == "SAMPLE_MESSAGE") {
      // 处理 message.payload
      // 可再通过 emitDeviceEvent 转发给 JS
    }
  }
};

C++向ArkTS侧发消息:

cpp 复制代码
m_deps->rnInstance.lock()->postMessageToArkTS("ANOTHER_MESSAGE", { key: "value" });

ArkTS侧监听C++消息:

typescript 复制代码
const unsubscribe = rnInstance.cppEventEmitter.subscribe("ANOTHER_MESSAGE", (value: object) => {
  // value = { key: "value" }
  unsubscribe();
});

五、核心要点总结

  • RN (JS) 与 ArkTS 之间通信,C++胶水层是桥梁,开发者主要关注两端API用法即可。
  • JS侧调用TurboModule方法 → C++胶水转发 → ArkTS实现能力 → Promise返回JS
  • ArkTS/Native侧发事件 → C++胶水转发 → DeviceEventEmitter/事件总线 → JS侧监听响应
  • 消息总线机制支持双向自定义事件/消息传递,适合复杂场景。

结语

只要理解"一切通信必经C++胶水层",并用好TurboModule/DeviceEventEmitter/消息总线,鸿蒙ArkTS与React Native JS的双端通信就会变得非常清晰、简单!

相关推荐
鹏程十八少1 天前
6. 2026金三银四 面试官最爱的 Binder:一次拷贝、Activity 启动流程,这篇全搞定
前端·面试·前端框架
Java后端的Ai之路1 天前
React 快速入门到精通教程:从零基础到能写项目
前端·react.js·前端框架
光影少年2 天前
中级前端需要会的东西都有那些?
前端·学习·前端框架
kyriewen2 天前
React Hooks原理:为什么不能写在if里?揭开Hook的“魔法”面纱
前端·react.js·前端框架
军军君012 天前
数字孪生监控大屏实战模板:交通云实时数据监控平台
前端·javascript·css·vue.js·typescript·前端框架·echarts
十一.3662 天前
012-014 对state的理解,初始化state,react中的事件绑定
前端·react.js·前端框架
禅思院2 天前
下篇:打造可观测的异步加载防御体系
前端·架构·前端框架
鹏程十八少2 天前
5. 2026金三银四 吐血整理!Android高级UI 自定义view面试25题,覆盖90%大厂考点
前端·面试·前端框架
nunumaymax3 天前
【第三章-react 应用(基于 react 脚手架)】
前端·react.js·前端框架