HarmonyOS 鸿蒙Next native线程队列中调用平台ArkTS的网络库 无法调通

发布于 1周前 作者 gougou168 来自 鸿蒙OS

HarmonyOS 鸿蒙Next native线程队列中调用平台ArkTS的网络库 无法调通 native线程无法绑定napi环境,无法在非js线程中调用napi。

2 回复

在ohos中,JavaScript函数通常只能在native插件的主线程里调用。如果native侧创建了其他线程,则不能从这些线程调用需要napi_env、napi_value或napi_ref的NAPI接口。当native侧有其他线程,并且需要根据这些线程的完成结果调用JavaScript函数时,这些线程必须与native侧的主线程进行通信,才能在主线程中调用JavaScript函数。

解决方案:使用线程安全函数。napi_threadsafe_function

// 指向napi_value js_cb
napi_ref cbObj = nullptr;
// 线程安全函数
napi_threadsafe_function tsfn;
// Native侧Value值
static int cValue;
// 回调

static void CallJs(napi_env env, napi_value js_cb, void * context, void * data)
{
    // 获取引用值
    napi_get_reference_value(env, cbObj, &js_cb);
    // 创建一个ArkTS number作为ArkTS function的入参。
    napi_value argv;
    napi_create_int32(env, cValue, &argv);
    napi_value result = nullptr;
    napi_call_function(env, nullptr, js_cb, 1, &argv, &result);
    napi_get_value_int32(env, result, &cValue);
}

// Native 主线程

static napi_value ThreadsTest(napi_env env, napi_callback_info info)
{
    OH_LOG_INFO(LOG_APP, "dwt CallJs 0");
    // 从ArkTS侧获取的参数的数量
    size_t argc = 1;
    napi_value js_cb, work_name;
    // 获取ArkTS参数
    napi_get_cb_info(env, info, &argc, &js_cb, nullptr, nullptr);
    // 指向napi_value js_cb 的 napi_ref cbObj
    napi_create_reference(env, js_cb, 1, &cbObj);
    // 通过UTF8编码的C字符串数据创建work_name
    napi_create_string_utf8(env, "Work Item", NAPI_AUTO_LENGTH, &work_name);
    // 创建线程安全函数
    napi_create_threadsafe_function(env, js_cb, NULL, work_name, 0, 1, NULL, NULL, NULL, CallJs, &tsfn);
    // 其他线程中调用线程安全函数
    std::thread t({
        // 可获取线程ID
        std::thread::id this_id = std::this_thread::get_id();
        napi_acquire_threadsafe_function(tsfn);
        napi_call_threadsafe_function(tsfn, NULL, napi_tsfn_blocking);
    });
    t.detach();
    return NULL;
}

static napi_value SetValue(napi_env env, napi_callback_info info) {
    // 从ArkTS侧获取的参数的数量
    size_t argc = 1;
    napi_value jsValue;
    // 获取ArkTS参数
    napi_get_cb_info(env, info, &argc, &jsValue, nullptr, nullptr);
    napi_get_value_int32(env, jsValue, &cValue);
    return NULL;
}

static napi_value GetValue(napi_env env, napi_callback_info info) {
    napi_value jsValue;
    napi_create_int32(env, cValue, &jsValue);
    return jsValue;
}

// index.d.ts
export const threadsTest: (func: object) => number;
export const setValue: (value: number) => number;
export const getValue: () => number;

// index.ets
import entry from ‘libentry.so’;
import hilog from ‘@ohos.hilog’;
@Entry
@Component
struct Index {
    @State value: number = 20;
    @State cnt: number = 10;
    build() {
        Row() {
            Column() {
                Text("value: " + this.value.toString())
                Button("setValue To Native")
                    .onClick(() => {
                        entry.setValue(this.value)
                    }).margin(10)
                Button("跨线程调用JS函数")
                    .onClick(() => {
                        entry.threadsTest((value: number) => {
                            value += 10
                            this.cnt += 20
                            hilog.info(0x0000, 'testTag', 'js callback value = ' + value + " , cnt = " + this.cnt);
                            return value
                        })
                    }).margin(10)
                Button("getValue From Native")
                    .onClick(() => {
                        this.value = entry.getValue()
                    }).margin(10)
            }
            .width('100%')
        }
        .height('100%')
    }
}

参考线程安全函数的实现

更多关于HarmonyOS 鸿蒙Next native线程队列中调用平台ArkTS的网络库 无法调通的实战系列教程也可以访问 https://www.itying.com/category-93-b0.html


在HarmonyOS(鸿蒙)Next环境中,如果在native线程队列中调用平台ArkTS的网络库时遇到无法调通的问题,这通常与线程环境、上下文切换或库依赖管理有关。以下是一些可能的原因及排查方向:

  1. 线程环境差异:Native线程与ArkTS线程可能运行在不同的执行环境中,包括内存管理、事件循环等。确保在native线程中正确初始化并管理ArkTS所需的上下文。

  2. 库依赖问题:检查ArkTS网络库是否已正确链接到native线程所在的可执行文件或库中。同时,验证所有依赖的库版本是否兼容。

  3. API调用限制:某些ArkTS API可能不支持在native线程中直接调用,需要封装或通过特定接口进行调用。查阅官方文档确认API的使用限制。

  4. 线程同步:如果在多线程环境中操作共享资源,确保使用适当的同步机制避免数据竞争和死锁。

  5. 错误处理和日志:增加详细的错误处理和日志记录,以便追踪调用失败的具体位置和原因。

如果问题依旧没法解决请联系官网客服,官网地址是:https://www.itying.com/category-93-b0.html

回到顶部