在 cpp 中嵌入 python:如何获取指向 python 函数(而不是 PyObject)的 C 指针?

Embedding python in cpp: How to get a C-pointer to a python function (not to a PyObject)?

本文关键字:python PyObject 指针 cpp 取指 何获 函数      更新时间:2023-10-16

我想为现有的C++应用程序启用用户python插件的编写。主应用程序将使用 boost::signals2 发出信号,触发用户 python 代码的执行。我设置了一个工作示例,其中调用了用户 python 模块中的初始化方法。用户可以订阅调用"连接"方法的信号:

import ctypes
CALLBACK_VOID_FUNC = CFUNCTYPE(None)
def myVoidHandler():
print("myVoidHandler (python)")
def initialize(myCHandle):
global myCDll, callback1
myCDll = ctypes.CDLL("CppEventsd",  handle = myCHandle)
[...]
callback1 = CALLBACK_VOID_FUNC(myVoidHandler)
myCDll.connect(b"readyToGo", callback1)
[...]

连接方法的签名:

extern "C" __declspec(dllexport) bool connect(char* signalName, void(*pF)())

现在 python 函数 myVoidHandler 可以从 cpp 调用,就像这样:

pF();

但更重要的是,pF可以连接到升压信号。

但是,我想避免用户经历这种情况(并免除我的支持请求,因为他们没有完全正确(,所以我想通过使用函数/插槽的命名约定从 cpp 端连接 python 函数。 我可以轻松地检索指向python回调的PyObjet指针:

PyObject *pModule, *pDict, *pVoidHandlerF;
pModule = PyImport_Import(pName);
pDict = PyModule_GetDict(pModule);
pVoidHandlerF = PyDict_GetItemString(pDict, "myVoidHandler");

但是我不知道如何从 cpp 端获取实际的函数指针(相当于 pF(!在python到Cpp的过渡中似乎正在进行一些转换。

有什么想法吗?

您可以创建自己的 C++ 函数,该函数使用 python C-API 调用 python 函数,而不是直接连接到 python 函数指针。下面是一个示例:

boost::signals2::signal<void(void)>& sig = [...];
PyObject* pVoidHandlerF = [...];
sig.connect([=]() {
PyObject_CallObject(pVoidHandlerF, NULL);
});

这允许您在C++端执行其他操作,例如,传递参数、返回值、检查错误等。