Qt5(C++)调用python3脚本(带输入输出参数)

17 篇文章 0 订阅
1 篇文章 0 订阅

本例中的环境: Qt 5 9.0(编译器为32位mingw或32位msvc均可),python3.5.4 -32bit

1 前期准备

1.1 准备测试用的python文件(取名不要为test.py即可

def show():
    print("hello!")
def sum(a,b):
    return (a+b)

def main():
    show()
    print(sum(1,2))
 
if __name__ == '__main__':
    main()

先使用python自带的IDLE运行一下,保证代码无误
在这里插入图片描述

1.2 创建一个空的qt项目,选择release构建方式,保证其运行无误。然后将准备好的python文件拷贝在工程目录中(和.exe平行)

在这里插入图片描述

2 将python库文件路径导入qt项目中

展示一下我的python库文件:
在这里插入图片描述
在这里插入图片描述
导入:
注意:看一下我的libs文件夹中并没有 -lpython35这个文件,但是也要这样写才行,请大家参照进行填写
在这里插入图片描述

3 开始调用python

main.cpp

#include <QApplication>
#include <QDebug>
#include <QString>

//解决 Python 和 Qt 的关键词 slots 冲突
#pragma push_macro("slots")
#undef slots
#include <Python.h>
#pragma pop_macro("slots")

using namespace std;

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    Py_Initialize();

    //如果初始化失败,返回
    if(!Py_IsInitialized())
    {
        qDebug()<<"Python init fail!";
        return a.exec();
    }
    //设置python文件路径
    PyRun_SimpleString("import sys");
    PyRun_SimpleString("sys.path.append('./')");

    //加载test.py文件
    PyObject *pModule = PyImport_ImportModule("testfile"); //注意python文件不要取名为test.py,会与python库文件冲突
    if(!pModule)
    {
        qDebug()<<"load pModule(testfile.py) fail!";
        return a.exec();
    }
    //加载函数show
    PyObject* pFun_show= PyObject_GetAttrString(pModule,"show");
    if(!pFun_show)
    {
        qDebug()<<"Get pFun_show(show) failed!";
        return a.exec();
    }
    //加载函数sum
    PyObject* pFun_sum= PyObject_GetAttrString(pModule,"sum");
    if(!pFun_sum)
    {
        qDebug()<<"Get pFun_sum(sum) failed!";
        return a.exec();
    }


    //调用函数show
    PyEval_CallObject(pFun_show,NULL);


    //调用函数sum
    PyObject* args=PyTuple_New(2);
    //【例 1】:输入参数:int  输出参数:int
    int iin1=1,iin2=2;  //输入参数赋值
    PyTuple_SetItem(args,0,Py_BuildValue("i",iin1));
    PyTuple_SetItem(args,1,Py_BuildValue("i",iin2));
    PyObject* pReturn1=PyEval_CallObject(pFun_sum,args);
    if(PyLong_Check(pReturn1))
    {
        int iout =PyLong_AsLong(pReturn1);
        qDebug()<<iout;  //打印输出
    }
    else
        qDebug()<<"return type is not Long!";

    //【例 2】:输入参数:float  输出参数:float
    float fin1=1.1,fin2=2.2;  //输入参数赋值
    PyTuple_SetItem(args,0,Py_BuildValue("f",fin1));
    PyTuple_SetItem(args,1,Py_BuildValue("f",fin2));
    PyObject* pReturn2=PyEval_CallObject(pFun_sum,args);
    if(PyFloat_Check(pReturn2))
    {
        float fout=PyFloat_AsDouble(pReturn2);
        qDebug()<<fout;  //打印输出
    }
    else
        qDebug()<<"return type is not Float!";

    //【例 3】:输入参数:QString  输出参数:QString
    QString strin1="Sun",strin2="tian"; //输入参数赋值
    PyTuple_SetItem(args,0,Py_BuildValue("s",strin1.toStdString().c_str()));
    PyTuple_SetItem(args,1,Py_BuildValue("s",strin2.toStdString().c_str()));
    PyObject* pReturn3=PyEval_CallObject(pFun_sum,args);
    if(PyUnicode_Check(pReturn3)){
        char* s=PyUnicode_AsUTF8(pReturn3);
        QString strout=QLatin1String(s);
        qDebug()<<strout;  //打印输出
    }
    else
        qDebug()<<"return type is not Unicode!";

    Py_Finalize(); //释放python
    return a.exec();
}

输出截图:
在这里插入图片描述

4 其中的坑:

(1)#include <Python.h>报错

Python\Python35-32\include\object.h:445: error: expected unqualified-id before ‘;’ token
PyType_Slot slots; / terminated by slot==0. */
原因: Python 和 Qt 的关键词 slots 冲突
解决:
#pragma push_macro(“slots”)
#undef slots
#include <Python.h>
#pragma pop_macro(“slots”)

(2)PyImport_ImportModule返回值为空

原因: py文件错误
解决:将python文件放在和qt可执行文件(.exe)平行的位置

(3)release\callPy_test.exe: -1: error: LNK1120: 15 个无法解析的外部命令

可能原因: python和qt编译器位数不对应
解决:采用位数一致的编译器(对于qt来说,和qt位数无关,只和编译器位数有关)

(4)要保证python文件自己运行不报错,否则qt调用会造成崩溃

(5)python文件不能取名为test.py

后记:使用Qt(c++)和python进行混合编程,可以发挥c++的速度优势和python的灵活和便捷优势,也可以使系统更好的实现模块化,是一种值得研究的技术。

  • 14
    点赞
  • 60
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
在 Windows 下使用 Qt C++ 调用 Python 有多种方法,以下是其中一种比较简单的方法: 1. 安装 Python 和 PyQt 首先,在 Windows 下安装 Python 和 PyQt。 2. 创建 Qt 项目 使用 Qt Creator 创建一个 Qt 项目,选择 C++ 应用程序。 3. 添加 Python 支持 在项目的 .pro 文件中添加以下内容: ``` CONFIG += link_pkgconfig PKGCONFIG += python-3.6 LIBS += -LC:/Python36/libs -lpython36 INCLUDEPATH += C:/Python36/include DEPENDPATH += C:/Python36/include ``` 其中,python-3.6 是你安装的 Python 版本号,C:/Python36 是 Python 的安装路径。 4. 创建 Python 脚本 在项目中创建一个 Python 脚本,例如 test.py,内容如下: ``` def add(a, b): return a + b ``` 5. 在 C++调用 PythonC++调用 Python 可以使用 Python.h 头文件和 Python 的 API。以下是一个简单的示例: ```cpp #include <Python.h> int main(int argc, char *argv[]) { Py_Initialize(); PyObject *pModule = PyImport_ImportModule("test"); if (pModule) { PyObject *pFunc = PyObject_GetAttrString(pModule, "add"); if (pFunc && PyCallable_Check(pFunc)) { PyObject *pArgs = PyTuple_New(2); PyTuple_SetItem(pArgs, 0, PyLong_FromLong(1)); PyTuple_SetItem(pArgs, 1, PyLong_FromLong(2)); PyObject *pResult = PyObject_CallObject(pFunc, pArgs); if (pResult) { long result = PyLong_AsLong(pResult); printf("result=%ld\n", result); Py_DECREF(pResult); } Py_DECREF(pArgs); } Py_DECREF(pFunc); } Py_DECREF(pModule); Py_Finalize(); return 0; } ``` 以上示例代码调用Python 脚本中的 add 方法,传入两个参数 1 和 2,输出结果 3。 注意事项: - 在调用Python API 后需要释放对象,避免内存泄漏。 - 实际项目中,可能需要使用 PyGILState_Ensure 和 PyGILState_Release 来保证线程安全。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值