QT设置回调函数给python调用——内置模块法

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

在C++中,可以使用函数指针或者函数对象来实现回调函数的建立和绑定。而在使用QT框架中,可以使用信号和槽机制来实现回调函数的功能。 1. 使用函数指针建立回调函数: 首先,定义一个函数指针类型,该函数指针类型与回调函数的函数签名相匹配。然后,将回调函数的地址赋值给函数指针变量。最后,在需要调用回调函数的地方,通过函数指针来调用回调函数。 示例代码如下: ```cpp // 定义回调函数的函数签名 typedef void (*CallbackFunc)(int); // 回调函数 void callback(int data) { // 回调函数的实现 // ... } int main() { // 建立回调函数 CallbackFunc func = callback; // 调用回调函数 func(10); return 0; } ``` 2. 使用函数对象建立回调函数: 可以定义一个函数对象类,并重载函数调用运算符(operator()),使其具有类似函数的行为。然后,将函数对象作为参数传递给需要回调的函数或者类的成员函数。 示例代码如下: ```cpp // 定义函数对象类 class Callback { public: void operator()(int data) { // 回调函数的实现 // ... } }; int main() { // 建立回调函数 Callback callback; // 调用回调函数 callback(10); return 0; } ``` 3. 使用QT的信号和槽机制建立回调函数: 在QT中,可以使用信号和槽机制来实现回调函数的功能。首先,在需要回调的类中定义一个信号,然后在另一个类中定义一个槽函数,并将信号与槽函数进行连接。当信号被触发时,槽函数会被自动调用。 示例代码如下: ```cpp // 定义一个发送信号的类 class Sender : public QObject { Q_OBJECT signals: void dataReady(int data); }; // 定义一个接收信号的类 class Receiver : public QObject { Q_OBJECT public slots: void onDataReady(int data) { // 回调函数的实现 // ... } }; int main() { // 创建发送信号的对象和接收信号的对象 Sender sender; Receiver receiver; // 连接信号和槽函数 QObject::connect(&sender, SIGNAL(dataReady(int)), &receiver, SLOT(onDataReady(int))); // 触发信号,调用回调函数 emit sender.dataReady(10); return 0; } ``` 以上是建立和绑定回调函数的几种常见方。具体选择哪种方取决于你的需求和代码结构。如果你有其他问题,请继续提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值