QT设置回调函数给python调用——内置模块法
1.QT 相关函数定义 和QT设置回调函数给python调用——参数法 - Henkk - 博客园 (cnblogs.com) 中的定义相同
// 实际的回调函数 void printValue(int value) { qDebug() << "printValue value: " << value; } int getValue(int value) { qDebug() << "getValue value: " << value; return value; } // 包装回调函数,使其成为Python可调用对象 PyObject* cprintValue(PyObject* self, PyObject* args) { int value; if (!PyArg_ParseTuple(args, "i", &value)) { return NULL; } printValue(value); return Py_None; } PyObject* cgetValue(PyObject* self, PyObject* args) { int value; if (!PyArg_ParseTuple(args, "i", &value)) { return NULL; } return Py_BuildValue("i", getValue(value)); } // 模块方法定义 // 结构体第三个参数 // METH_VARARGS:方法接受可变数量的参数。 // METH_KEYWORDS:方法接受关键字参数。 // METH_NOARGS:方法不接受参数。 // METH_O:方法接受一个对象参数。 static PyMethodDef CallbackMethods[] = { {"cprintValue", cprintValue, METH_VARARGS, "printValue callback function"}, {"cgetValue", cgetValue, METH_VARARGS, "getValue callback function"}, {NULL, NULL, 0, NULL} //这一行为结束的标识 }; // 模块定义 static struct PyModuleDef CallbackModule = { PyModuleDef_HEAD_INIT, "callback", // 模块名称 NULL, // 模块文档描述 -1, // 模块状态 CallbackMethods, //模块中的函数和方法 nullptr, //指向一个 PyModuleDef_Slot 结构体数组的指针,用于定义模块的特殊属性 nullptr, //指向一个函数指针,用于遍历模块对象的函数 nullptr, //指向一个函数指针,用于清除模块对象的函数 nullptr //指向一个函数指针,用于释放模块对象的函数 }; // 模块初始化函数 PyMODINIT_FUNC PyInit_callback(void) { PyObject *m = PyModule_Create(&CallbackModule); if(m == NULL){ qDebug("PyInit_callback failed!"); return NULL; } return m; }
2. python 相关函数定义
# python_script_2.py import callback def printValue(): callback.cprintValue(45) def getValue(): callback.cgetValue(256)
3. QT 测试
int main(int argc, char *argv[]) { QCoreApplication app(argc, argv); // 设置Python解释器的路径 wchar_t *program = Py_DecodeLocale("python38_64", nullptr); Py_SetProgramName(program); Py_SetPath(L"D:\\Python\\Python38_64;D:\\Python\\Python38_64\\DLLs;D:\\Python\\Python38_64\\lib"); // 添加自定义模块 PyImport_AppendInittab("callback", PyInit_callback); // 初始化Python解释器 Py_Initialize(); PyRun_SimpleString("import sys"); PyRun_SimpleString("sys.path.append('./')"); PyObject *demo = PyImport_ImportModule("python_script_2"); if(demo == NULL){ qInfo() << "demo is nullptr!"; PyErr_Print(); return -1; } PyObject *pyPrintValue = PyObject_GetAttrString(demo,"printValue"); if(pyPrintValue == NULL){ qInfo() << "pyPrintValue is nullptr!"; return -1; } PyObject* pReturnValue; PyObject* args = PyTuple_New(0); pReturnValue = PyObject_CallObject(pyPrintValue, args); PyObject *pyGetValue = PyObject_GetAttrString(demo,"getValue"); if(pyGetValue == NULL){ qInfo() << "pgetValue is nullptr!"; return -1; } PyObject* pReturnValue2; PyObject* args2 = PyTuple_New(0); pReturnValue2 = PyObject_CallObject(pyGetValue, args2); Py_DECREF(demo); // 结束Python解释器 Py_Finalize(); return app.exec(); }
说明:
PyImport_AppendInittab 函数通常在 Python 解释器初始化之前调用。它的作用是注册一个 C 函数,以便在 Python 解释器启动时初始化一个新的内置模块。因此,你需要在 Python 解释器初始化之前调用 PyImport_AppendInittab 函数,以确保你的自定义模块能够正确初始化
相当于使用C++/QT写一个内置的python模块在初始化之前内置到解释器,后面的模块都可以使用。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本