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;
// ...
并确保对象在回调触发时处于活动状态。
我正在使用 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;
// ...
并确保对象在回调触发时处于活动状态。