1.QT 相关函数定义 和 QT设置回调函数给python调用——参数法中的定义相同如下:
// 实际的回调函数
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模块在初始化之前内置到解释器,后面的模块都可以使用。
标签:初始化,解释器,内置,QT,callback,python,Python From: https://blog.csdn.net/weixin_39344648/article/details/141227878