Qt C++ 调用 Python——常用操作的封装 -- 加载模块、获取函数、实例化类、获取类的方法和调用 Python 方法

Qt C++ 调用 Python——常用操作的封装 – 加载模块、获取函数、实例化类、获取类的方法和调用 Python 方法

将常见的交互操作(如加载模块、获取函数、实例化类、获取类方法和调用 Python 方法)封装起来,可以极大地简化 Qt C++ 和 Python 之间的交互。以下详细介绍了如何封装这些操作,并提供了对应的示例代码。

1. 加载模块

加载 Python 模块是使用 Python 功能的第一步。以下函数演示了如何在 Qt C++ 中加载 Python 模块:

 * @brief LoadPythonModule 获取加载模块对象
 * @param moduleName      模块名称
 * @param scriptPath      脚本路径
 * @return 加载的模块对象
 */
PyObject* LoadPythonModule(const QString &moduleName, const QString &scriptPath)
{
    if (!Py_IsInitialized())
    {
        qDebug() << "Python 解释器未初始化!";
        return nullptr;
    }

    // 添加 Python 模块路径 将路径添加到sys.path
    QString pyCode = QString("import sys; sys.path.append('%1')").arg(scriptPath);

    // 将QString转换为C风格字符串
    QByteArray pyCodeBytes = pyCode.toUtf8();
    const char* pyCodeCStr = pyCodeBytes.constData();

    PyRun_SimpleString(pyCodeCStr);

    // 加载Python模块
    PyObject* pModuleObject = PyImport_ImportModule(moduleName.toStdString().c_str());
    if (!pModuleObject)
    {
        PyErr_Print();
        qDebug() << "加载模块失败!";
        return nullptr;
    }
    else
    {
        qDebug() << moduleName + "模块加载成功!";
    }

    return pModuleObject;
}

2. 获取模块中的函数

从已加载的模块中获取特定函数:

/**
 * @brief GetFuncObject 获取函数对象
 * @param moduleObject  已加载的模块对象
 * @param funcName      要获取的函数的名称
 * @return  函数对象
 */
PyObject* GetPythonFunction(PyObject *moduleObject, const QString &funcName)
{
    if (!moduleObject)
    {
        qDebug() << "模块未加载,无法调用函数!";
        return nullptr;
    }

    PyObject* pFuncObject = PyObject_GetAttrString(moduleObject, funcName.toStdString().c_str());

    if (!pFuncObject)
    {
        PyErr_Print();
        qDebug() << "获取函数失败!";
        return nullptr;
    }

    if (!PyCallable_Check(pFuncObject)) // 检查函数对象是否可调用
    {
        PyErr_Print();
        qDebug() << "函数不可调用!";
        Py_DECREF(pFuncObject);  // 释放对象
        return nullptr;
    }
    else
    {
        qDebug() << funcName + "函数加载成功";
    }

    return pFuncObject;
}

3. 获取 Python 类的实例

创建并返回指定 Python 类的实例:

/**
  * @brief GetPythonClassInstance 获取 Python 类的实例
  * @param moduleObject 已加载的 Python 模块对象
  * @param className 要实例化的类的名称
  * @param args      传递给类构造函数的参数(可选)- args 可以是元组
  * @param kwargs    传递给类构造函数的字典(可选)- 传入字典 args 需要为空
  * @return Python   类的实例
  */
PyObject* GetPythonClassInstance(PyObject *moduleObject, const QString &className, PyObject *args, PyObject *kwargs)
{
if (!moduleObject)
    {
        qDebug() << "模块未加载,无法获取类实例!";
        return nullptr;
    }

    PyObject* pClassObject = PyObject_GetAttrString(moduleObject, className.toStdString().c_str());

    if (!pClassObject)
    {
        PyErr_Print();
        qDebug() << "获取类失败!";
        return nullptr;
    }

    if (!PyCallable_Check(pClassObject))
    {
        PyErr_Print();
        qDebug() << "类不可调用!";
        Py_DECREF(pClassObject);  // 释放对象
        return nullptr;
    }

    // 调用类构造函数,传递 args 和 kwargs 参数
    PyObject* pInstance = PyObject_Call(pClassObject, args ? args : PyTuple_New(0), kwargs);
    Py_DECREF(pClassObject);  // 释放类对象

    if (!pInstance)
    {
        PyErr_Print();
        qDebug() << "实例化类失败!";
        return nullptr;
    }

    qDebug() << className + "类实例化成功";
    return pInstance;
}

4. 获取类的方法

从 Python 类实例中获取特定方法:

/**
 * @brief GetClassMethod 获取类的方法
 * @param classInstance Python 类的实例对象
 * @param methodName 要获取的方法的名称
 * @return 类的方法
 */
PyObject* GetClassMethod(PyObject* classInstance, const QString& methodName)
{
    if (!classInstance)
    {
        qDebug() << "类实例未创建,无法获取方法!";
        return nullptr;
    }

    PyObject* pMethodObject = PyObject_GetAttrString(classInstance, methodName.toStdString().c_str());

    if (!pMethodObject)
    {
        PyErr_Print();
        qDebug() << "获取方法失败!";
        return nullptr;
    }

    if (!PyCallable_Check(pMethodObject))
    {
        PyErr_Print();
        qDebug() << "方法不可调用!";
        Py_DECREF(pMethodObject);  // 释放对象
        return nullptr;
    }

    qDebug() << methodName + "方法加载成功";
    return pMethodObject;
}

5. 调用 Python 函数

执行 Python 函数并返回结果:

/**
 * @brief CallPythonFunction 调用 Python 函数
 * @param funcObject  要调用的函数对象
 * @param args        传递给函数的参数元组(可选)     - args 可以是元组
 * @param kwargs      传递给函数的关键字参数字典(可选)- 传入字典 args 需要为空
 * @return 函数调用的结果
 */
PyObject* CallPythonFunction(PyObject* funcObject, PyObject* args = nullptr, PyObject* kwargs = nullptr)
{
    if (!funcObject)
    {
        qDebug() << "函数对象无效,无法调用函数!";
        return nullptr;
    }

    if (!PyCallable_Check(funcObject))
    {
        PyErr_Print();
        qDebug() << "函数对象不可调用!";
        return nullptr;
    }

    // 调用函数,传递 args 和 kwargs 参数
    PyObject* pResult = PyObject_Call(funcObject, args ? args : PyTuple_New(0), kwargs);

    if (!pResult)
    {
        PyErr_Print();
        qDebug() << "调用函数失败!";
        return nullptr;
    }

    qDebug() << "函数调用成功";
    return pResult;
}
  • 6
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值