C++ class 中的成功回调 Emscripten FETCH API

Success callback Emscripten FETCH API in C++ class

我正在使用 WebAssembly 并尝试从 C++ 发出 HTTPS 请求。我看过Emscripten FETCH API的解决方案,并尝试使用它。

为了测试它,我创建了一个 Test class,我在其中发送这样的请求:

void Test::sendRequest() {
    emscripten_fetch_attr_t attr;
    emscripten_fetch_attr_init(&attr);
    strcpy(attr.requestMethod, "GET");
    attr.attributes = EMSCRIPTEN_FETCH_LOAD_TO_MEMORY;
    attr.onsuccess = &Test::onSuccess;
    attr.onerror = &Test::onError;
    emscripten_fetch(&attr, "http://127.0.0.1:5000/");
}

我的 onSuccess 回调看起来像这样:

void Test::onSuccess(struct emscripten_fetch_t *fetch) {
    printf("Finished downloading %llu bytes from URL %s.\n", fetch->numBytes, fetch->url);
    setText(QString::fromUtf8(fetch->data));
    emscripten_fetch_close(fetch); // Free data associated with the fetch.
}

但是,当我尝试编译时出现错误提示:

error: assigning to 'void (*)(struct emscripten_fetch_t *)' from incompatible type 'void
  (Test::*)(struct emscripten_fetch_t *)'
attr.onsuccess = &Test::onSuccess;
                 ^~~~~~~~~~~~~~~~

看来我不能将回调函数放在class中,但我需要访问class才能使用响应修改实例的文本属性。

我尝试用单例模式定义测试 class 并从 class 中删除回调函数。使用这种方法,我可以修改文本属性以获得 class 的唯一实例,但如果可能的话,我想直接将回调函数放在 class 中。

不能直接使用非静态成员函数作为回调。

但是,大多数回调接口在某处有一个 "user data" 字段用于与发起者通信。

emscripten_fetch_attr_t 有一个 void* userData 成员,您可以在其中存储您想要的任何指针。
该指针在回调参数中作为 userData 传递,您只需将其转换回正确的类型即可。

所以你可以使用一个自由函数作为包装回调,对象为"user data":

void onSuccess(struct emscripten_fetch_t *fetch) {
    auto test = static_cast<Test*>(fetch->userData);
    test->onSuccess(fetch);
}

void Test::sendRequest() {
    emscripten_fetch_attr_t attr;
    emscripten_fetch_attr_init(&attr);
    strcpy(attr.requestMethod, "GET");
    attr.attributes = EMSCRIPTEN_FETCH_LOAD_TO_MEMORY;
    attr.userData = this;
    attr.onsuccess = onSuccess;
    // ...

并确保对象在回调触发时处于活动状态。