Henkk

导航

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模块在初始化之前内置到解释器,后面的模块都可以使用。

posted on 2024-07-09 14:10  Henkk  阅读(1)  评论(0编辑  收藏  举报