- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
假设我们有这样一种 C 风格的 API:
void register_callback(target, callback_function);
,其中target
是某种对象,例如服务器,和 call_back
必须是函数指针。
现在我们要适应(更准确地说,添加一个包装层。我们无法访问底层 C 代码,因为它是第三方库)这个 API 到 C++。我们需要 OOP,所以我们将目标变成一个类:
class Target {
Target(): target() { register_callback(target, callback_function_member) }
private:
target_t target;
void callback_function_member (param_t parameter) { /* a set of things to do with the members */ }
// other members
}
我们可能希望像上面那样使回调成为成员函数,因为它(在某些情况下)与实例紧密相关。但这行不通。没有办法将成员函数调整为函数指针(参见,例如 Pointers to Member Functions )。
而且我在网上找到的所有解决方案(包括上一个)都只处理只有一个实例负责工作的情况。在那里我们可以专门编写一个包装函数(参见 1 )或一个包装类的静态函数。
问题是,如果我们想要多个实例,并且动态创建它们怎么办?您可以为每个实例编写一个包装函数或类,例如
Target *pTarget;
void wrapper(param_t parameter) {
pTarget->callback_function_member(parameter);
}
int main() {
Target myTarget();
pTarget = &myTarget();
myTarget.register(wrapper);
// some work
return 0;
}
但是您不能为动态创建的每个实例手动编写一个函数。而且似乎不可能将任何类型的信息传递到全局函数中(例如,将实例存储在全局容器中,但是调用时如何传递索引?)。
std::bind
接近解决方案(我不知道它的存在),但关键问题是 std::bind
的返回值不能转换为函数指针。使用 target<type_of_function_pointer>()
的 std::function
给我一个nil
,因为类型不匹配。当我假装它是一个函数指针并直接传递它时,发生了段错误。
这到底有可能吗?还是我的目标被误导了?处理这个问题的规范方法是什么?
最佳答案
如果您确实需要 C 风格的回调,即通过常规函数指针调用目标回调的回调,那么处理此问题的规范方法是将实际回调实现为“常规”(非成员或静态成员)函数并将对象指针作为额外参数传递给它
class SomeClass {
...
void member_callback(int param1, double param2) { ... }
static void static_callback_wrapper(int param1, double param2, void *user_param) {
return static_cast<SomeClass *>(user_param)->member_callback(param1, params2);
}
};
现在,如果你有一个对象
SomeClass some_object;
并且您想将回调与此对象一起使用,在某些外部算法中,您传递一个指向 SomeClass::static_callback_wrapper
的指针作为回调,以及一个指向 some_object
作为该算法的“void *
opaque user-specified data”。当然,该算法的实现必须考虑到用户指定的数据 - 它必须将您从外部传递的 void *
指针转发回您的回调。
例如,这是 pthreads 库中遵循的约定 pthread_create
(http://man7.org/linux/man-pages/man3/pthread_create.3.html)。您传递给该函数的回调是 void *(*start_routine) (void *)
,“用户指定的数据”通过最后一个 void *arg
参数传递.线程函数将作为 start_routine(arg)
调用,从而将用户指定的指针转发回回调。
非标准 qsort_r
函数 (http://linux.die.net/man/3/qsort_r) 中使用了相同的约定。基本上,将这种“不透明的用户指定数据”转发机制包含到每个使用 C 风格回调的实现中是一个很好的编程实践。
所有这些适用于您确实需要符合 C 风格回调接口(interface)的情况。在纯 C++ 中,您还有其他更灵活的机会。您可以用 std::function
对象表示回调(作为一种可能性),并简单地将成员函数的隐式 this
参数绑定(bind)到您要用于回调,从而将它们变成“普通”函数。
void some_algorithm(std::function<void ()> callback) {
...
callback();
...
callback();
...
}
class SomeClass {
void foo() {}
};
int main() {
SomeClass object;
some_algorithm(std::bind(&SomeClass::foo, &object));
}
关于c++ - 是否可以将 C 风格的 api 回调/处理程序适应 C++ 以进行类相关的动态创建?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35488919/
对于 Metal ,如果对主纹理进行 mipmap 处理,是否还需要对多采样纹理进行 mipmap 处理?我阅读了苹果文档,但没有得到任何相关信息。 最佳答案 Mipmapping 适用于您将从中
我正在使用的代码在后端 Groovy 代码中具有呈现 GSP(Groovy 服务器页面)的 Controller 。对于前端,我们使用 React-router v4 来处理路由。我遇到的问题是,通过
我们正在 build 一个巨大的网站。我们正在考虑是在服务器端(ASP .Net)还是在客户端进行 HTML 处理。 例如,我们有 HTML 文件,其作用类似于用于生成选项卡的模板。服务器端获取 HT
我正在尝试将图像加载到 void setup() 中的数组中,但是当我这样做时出现此错误:“类型不匹配,'processing .core.PImage' does not匹配“processing.
我正在尝试使用其私有(private)应用程序更新 Shopify 上的客户标签。我用 postman 尝试过,一切正常,但通过 AJAX,它带我成功回调而不是错误,但成功后我得到了身份验证链接,而不
如何更改我的 Processing appIconTest.exe 导出的默认图标在窗口中的应用程序? 默认一个: 最佳答案 经过一些研究,我能找到的最简单的解决方案是: 进入 ...\process
我在 Processing 中做了一个简单的小游戏,但需要一些帮助。我有一个 mp3,想将它添加到我的应用程序中,以便在后台循环运行。 这可能吗?非常感谢。 最佳答案 您可以使用声音库。处理已经自带
我有几个这样创建的按钮: 在 setup() PImage[] imgs1 = {loadImage("AREA1_1.png"),loadImage("AREA1_2.png"),loadImage
我正在尝试使用 Processing 创建一个多人游戏,但无法弄清楚如何将屏幕分成两个以显示玩家的不同情况? 就像在 c# 中一样,我们有Viewport leftViewport,rightView
我一直在尝试使用 Moore 邻域在处理过程中创建元胞自动机,到目前为止非常成功。我已经设法使基本系统正常工作,现在我希望通过添加不同的功能来使用它。现在,我检查细胞是否存活。如果是,我使用 fill
有没有办法用 JavaScript 代码检查资源使用情况?我可以检查脚本的 RAM 使用情况和 CPU 使用情况吗? 由于做某事有多种方法,我可能会使用不同的方法编写代码,并将其保存为两个不同的文件,
我想弄清楚如何处理这样的列表: [ [[4,6,7], [1,2,4,6]] , [[10,4,2,4], [1]] ] 这是一个整数列表的列表 我希望我的函数将此列表作为输入并返回列表中没有重复的整
有没有办法在不需要时处理 MethodChannel/EventChannel ?我问是因为我想为对象创建多个方法/事件 channel 。 例子: class Call { ... fields
我有一个关于在 Python3 中处理 ConnectionResetError 的问题。这通常发生在我使用 urllib.request.Request 函数时。我想知道如果我们遇到这样的错误是否可
我一直在努力解决这个问题几个小时,但无济于事。代码很简单,一个弹跳球(粒子)。将粒子的速度初始化为 (0, 0) 将使其保持上下弹跳。将粒子的初始化速度更改为 (0, 0.01) 或任何十进制浮点数都
我把自己弄得一团糟。 我想在我的系统中添加 python3.6 所以我决定在我的 Ubuntu 19.10 中卸载现有的。但是现在每次我想安装一些东西我都会得到这样的错误: dpkg: error w
我正在努力解决 Rpart 包中的 NA 功能。我得到了以下数据框(下面的代码) Outcome VarA VarB 1 1 1 0 2 1 1 1
我将 Java 与 JSF 一起使用,这是 Glassfish 3 容器。 在我的 Web 应用程序中,我试图实现一个文件(图像)管理系统。 我有一个 config.properties我从中读取上传
所以我一直在Processing工作几个星期以来,虽然我没有编程经验,但我已经转向更复杂的项目。我正在编写一个进化模拟器,它会产生具有随机属性的生物。 最终,我将添加复制,但现在这些生物只是在屏幕上漂
有人知道 Delphi 2009 对“with”的处理有什么不同吗? 我昨天解决了一个问题,只是将“with”解构为完整引用,如“with Datamodule、Dataset、MainForm”。
我是一名优秀的程序员,十分优秀!