当前位置:   article > 正文

[HarmonyOS应用开发]使用NAPI实现C与ArkTS交互_arkts 与c文件

arkts 与c文件

使用Node-API实现跨语言交互

  • ArkTS侧:实现C++方法的结果返回,以及实现其回调函数。

  • Native侧:.cpp文件,实现结果返回函数、回调函数以及设置回调数据。需要提供注册lib库的名称,并在注册回调方法中定义接口的映射关系,即Native方法及对应的ArkTS接口名称等。

与NAPI开发的相关目录介绍

  • ntry > src > main > cpp > types:用于存放C++的API接口描述文件。

  • entry > src > main > cpp > types > libentry > index.d.ts:描述C++ API接口行为,如接口名、入参、返回参数等。

  • entry > src > main > cpp > types > libentry > oh-package.json5:配置.so三方包声明文件的入口及包名。

  • entry > src > main > cpp > CMakeLists.txt:C++源码编译配置文件,提供CMake构建脚本。

  • entry > src > main > cpp > xxx.cpp:定义C++ API接口的文件。

Native侧方法的实现

  • 1. 设置模块注册信息

ArkTS侧import native模块时,会加载其对应的so。加载so时,首先会调用napi_module_register方法,将模块注册到系统中,并调用模块初始化函数。

napi_module有两个关键属性:一个是.nm_register_func,定义模块初始化函数;另一个是.nm_modname,定义模块的名称,也就是ArkTS侧引入的so库的名称,模块系统会根据此名称来区分不同的so。

  1. // 准备模块加载相关信息,将上述Init函数与本模块名等信息记录下来。
  2. static napi_module testModule = {
  3. .nm_version = 1,
  4. .nm_flags = 0,
  5. .nm_filename = nullptr,
  6. .nm_register_func = Init,
  7. .nm_modname = "entry",
  8. .nm_priv = ((void*)0),
  9. .reserved = { 0 },
  10. };
  11. // 加载so时,该函数会自动被调用,将上述testModule模块注册到系统中。
  12. extern "C" __attribute__((constructor)) void RegisterEntryModule(void)
  13. {
  14. napi_module_register(&testModule);
  15. }
  • 2. 模块初始化

实现ArkTS接口与C++接口的绑定和映射。

  1. EXTERN_C_START
  2. static napi_value Init(napi_env env, napi_value exports)
  3. {
  4. napi_property_descriptor desc[] = {
  5. { "resultMsg", nullptr, resultMsg, nullptr, nullptr, nullptr, napi_default, nullptr },
  6. {"setOnCallBack", nullptr, setOnCallBack, nullptr, nullptr, nullptr, napi_default, nullptr},
  7. {"nativeToTSCallBack", nullptr, nativeToTSCallBack, nullptr, nullptr, nullptr, napi_default, nullptr}
  8. };
  9. napi_define_properties(env, exports, sizeof(desc) / sizeof(desc[0]), desc);
  10. return exports;
  11. }
  12. EXTERN_C_END
  • 3. 在index.d.ts文件中,提供JS侧的接口方法。
  1. export const resultMsg: (a: number, b: number) => number;
  2. export const setOnCallBack: (cbFn: (progress: number) => void) => void;
  3. export const nativeToTSCallBack: (a: number) => void;
  • 4. 在oh-package.json5文件中将index.d.ts与cpp文件关联起来。
  1. {
  2. "name": "libentry.so",
  3. "types": "./index.d.ts",
  4. "version": "",
  5. "description": "Please describe the basic information."
  6. }
  • 5. 在CMakeLists.txt文件中配置CMake打包参数。
  1. # the minimum version of CMake.
  2. cmake_minimum_required(VERSION 3.4.1)
  3. project(TestC)
  4. set(NATIVERENDER_ROOT_PATH ${CMAKE_CURRENT_SOURCE_DIR})
  5. if(DEFINED PACKAGE_FIND_FILE)
  6. include(${PACKAGE_FIND_FILE})
  7. endif()
  8. include_directories(${NATIVERENDER_ROOT_PATH}
  9. ${NATIVERENDER_ROOT_PATH}/include)
  10. # 添加名为entry的库
  11. add_library(entry SHARED napi_init.cpp)
  12. # 构建此可执行文件需要链接的库
  13. target_link_libraries(entry PUBLIC libace_napi.z.so)

实现Native的结果返回

  • 1.Native侧的函数代码为:
  1. /**
  2. * 返回测试
  3. */
  4. static napi_value resultMsg(napi_env env, napi_callback_info info)
  5. {
  6. size_t argc = 2;
  7. napi_value args[2] = {nullptr};
  8. napi_get_cb_info(env, info, &argc, args , nullptr, nullptr);
  9. napi_valuetype valuetype0;
  10. napi_typeof(env, args[0], &valuetype0);
  11. napi_valuetype valuetype1;
  12. napi_typeof(env, args[1], &valuetype1);
  13. double value0;
  14. napi_get_value_double(env, args[0], &value0);
  15. double value1;
  16. napi_get_value_double(env, args[1], &value1);
  17. napi_value sum;
  18. napi_create_double(env, value0 + value1, &sum);
  19. return sum;
  20. }
  • 2. ArkTS侧调用Native侧的结果返回函数“resultMsg”
  1. Button(this.resultMsg)
  2. .fontSize(20)
  3. .fontWeight(FontWeight.Bold)
  4. .onClick(() => {
  5. this.add += 1;
  6. this.resultMsg = `C函数的结果返回,返回结果为:${testNapi.resultMsg(this.add, 3)}`
  7. })

实现Native的数据回调

  • 1.Native侧的设置回调函数代码为:
  1. /**
  2. * 设置回调方法
  3. */
  4. static napi_value setOnCallBack(napi_env env, napi_callback_info info) {
  5. size_t argc = 1;
  6. napi_value args[1] = {nullptr};
  7. napi_get_cb_info(env, info, &argc, args, nullptr, nullptr);
  8. asyncContext->env = env;
  9. napi_create_reference(env, args[0], 1, &asyncContext->callbackRef);
  10. return nullptr;
  11. }
  • 2.Native侧的触发回调函数代码为:
  1. /**
  2. * 测试数据回调到TS端展现出来
  3. */
  4. static napi_value nativeToTSCallBack(napi_env env, napi_callback_info info) {
  5. size_t argc = 1;
  6. napi_value args[1] = {nullptr};
  7. napi_get_cb_info(env, info, &argc, args, nullptr, nullptr);
  8. napi_valuetype valuetype0;
  9. napi_typeof(env, args[0], &valuetype0);
  10. int value0;
  11. napi_get_value_int32(env, args[0], &value0);
  12. asyncContext->progress = value0;
  13. std::thread callBackThread(callBackTask);
  14. callBackThread.detach();
  15. return nullptr;
  16. }
  • 3.ArkTS侧实现回调的代码:
  1. aboutToAppear(): void {
  2. testNapi.setOnCallBack((data: number) => {
  3. this.callBackMsg = `C函数的结果回调,回调结果为:${data}`
  4. });
  5. }
  • 4.ArkTS侧触发C++函数进行回调的代码:
  1. Button('函数回调测试')
  2. .fontSize(20)
  3. .fontWeight(FontWeight.Bold)
  4. .onClick(() => {
  5. testNapi.nativeToTSCallBack(1)
  6. })
  7. .margin({'top': 50})

Native侧完整代码:

  1. #include "napi/native_api.h"
  2. #include <thread>
  3. #include <uv.h>
  4. /**
  5. * 返回测试
  6. */
  7. static napi_value resultMsg(napi_env env, napi_callback_info info)
  8. {
  9. size_t argc = 2;
  10. napi_value args[2] = {nullptr};
  11. napi_get_cb_info(env, info, &argc, args , nullptr, nullptr);
  12. napi_valuetype valuetype0;
  13. napi_typeof(env, args[0], &valuetype0);
  14. napi_valuetype valuetype1;
  15. napi_typeof(env, args[1], &valuetype1);
  16. double value0;
  17. napi_get_value_double(env, args[0], &value0);
  18. double value1;
  19. napi_get_value_double(env, args[1], &value1);
  20. napi_value sum;
  21. napi_create_double(env, value0 + value1, &sum);
  22. return sum;
  23. }
  24. typedef struct CallbackContext {
  25. napi_env env = nullptr;
  26. napi_ref callbackRef = nullptr;
  27. int progress = 0;
  28. } CallbackContext;
  29. auto asyncContext = new CallbackContext();
  30. // 回调ts侧函数,将进度信息通知到ts侧
  31. static void callTS(napi_env env, napi_value jsCb, void *context, void *data) {
  32. CallbackContext *arg = (CallbackContext *)data;
  33. napi_value progress;
  34. napi_create_int32(arg->env, arg->progress, &progress);
  35. napi_call_function(arg->env, nullptr, jsCb, 1, &progress, nullptr);
  36. }
  37. void callBackTask() {
  38. // 创建线程安全函数
  39. napi_value workName;
  40. napi_create_string_utf8(asyncContext->env, "callBack", NAPI_AUTO_LENGTH, &workName);
  41. napi_value jsCb;
  42. napi_get_reference_value(asyncContext->env, asyncContext->callbackRef, &jsCb);
  43. napi_threadsafe_function tsfn;
  44. napi_create_threadsafe_function(asyncContext->env, jsCb, nullptr, workName, 0, 1, nullptr, nullptr, nullptr, callTS,
  45. &tsfn);
  46. while (asyncContext && asyncContext->progress < 10) {
  47. asyncContext->progress += 1;
  48. napi_acquire_threadsafe_function(tsfn);
  49. napi_call_threadsafe_function(tsfn, (void *)asyncContext, napi_tsfn_blocking);
  50. std::this_thread::sleep_for(std::chrono::milliseconds(100));
  51. }
  52. };
  53. /**
  54. * 设置回调方法
  55. */
  56. static napi_value setOnCallBack(napi_env env, napi_callback_info info) {
  57. size_t argc = 1;
  58. napi_value args[1] = {nullptr};
  59. napi_get_cb_info(env, info, &argc, args, nullptr, nullptr);
  60. asyncContext->env = env;
  61. napi_create_reference(env, args[0], 1, &asyncContext->callbackRef);
  62. return nullptr;
  63. }
  64. /**
  65. * 测试数据回调到TS端展现出来
  66. */
  67. static napi_value nativeToTSCallBack(napi_env env, napi_callback_info info) {
  68. size_t argc = 1;
  69. napi_value args[1] = {nullptr};
  70. napi_get_cb_info(env, info, &argc, args, nullptr, nullptr);
  71. napi_valuetype valuetype0;
  72. napi_typeof(env, args[0], &valuetype0);
  73. int value0;
  74. napi_get_value_int32(env, args[0], &value0);
  75. asyncContext->progress = value0;
  76. std::thread callBackThread(callBackTask);
  77. callBackThread.detach();
  78. return nullptr;
  79. }
  80. EXTERN_C_START
  81. static napi_value Init(napi_env env, napi_value exports)
  82. {
  83. napi_property_descriptor desc[] = {
  84. { "resultMsg", nullptr, resultMsg, nullptr, nullptr, nullptr, napi_default, nullptr },
  85. {"setOnCallBack", nullptr, setOnCallBack, nullptr, nullptr, nullptr, napi_default, nullptr},
  86. {"nativeToTSCallBack", nullptr, nativeToTSCallBack, nullptr, nullptr, nullptr, napi_default, nullptr}
  87. };
  88. napi_define_properties(env, exports, sizeof(desc) / sizeof(desc[0]), desc);
  89. return exports;
  90. }
  91. EXTERN_C_END
  92. // 准备模块加载相关信息,将上述Init函数与本模块名等信息记录下来。
  93. static napi_module testModule = {
  94. .nm_version = 1,
  95. .nm_flags = 0,
  96. .nm_filename = nullptr,
  97. .nm_register_func = Init,
  98. .nm_modname = "entry",
  99. .nm_priv = ((void*)0),
  100. .reserved = { 0 },
  101. };
  102. // 加载so时,该函数会自动被调用,将上述testModule模块注册到系统中。
  103. extern "C" __attribute__((constructor)) void RegisterEntryModule(void)
  104. {
  105. napi_module_register(&testModule);
  106. }

应用界面

参考:

使用Node-API实现跨语言交互开发流程-使用Node-API实现跨语言交互-代码开发-NDK开发-开发 | 华为开发者联盟 (huawei.com)

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/从前慢现在也慢/article/detail/745981
推荐阅读
相关标签
  

闽ICP备14008679号