文章目录
显示
作者:虚坏叔叔
博客:https://xuhss.com
早餐店不会开到晚上,想吃的人早就来了!?
C++调用python函数传递list参数并获取返回
一、C++调用python无参数返回值函数
在test.py中添加函数:
print('test.py')
def Main():
print("Python Main")
在c++中调用这个函数
// C++调用Python函数
PyObject* Main = PyObject_GetAttrString(m, "Main");
if (Main && PyCallable_Check(Main)) {
// 函数对象和参数 返回对象
if (!PyObject_CallObject(Main, 0))
{
throw exception("PyObject_CallObject Failed");
}
}
Py_XDECREF(Main);
運行:
可以看到成功調用了。
二、C++调用python带参数和返回值的函数
在test.py
中添加函数TestFun
:
print('test.py')
def Main():
print("Python Main")
def TestFun(lis):
print("In Python", lis)
return [1,2,3,4,5,6,7,8,9,10]
在C++中調用
// C++调用Python函数
PyObject* Main = PyObject_GetAttrString(m, "Main");
if (Main && PyCallable_Check(Main)) {
// 函数对象和参数 返回对象
if (!PyObject_CallObject(Main, 0))
{
throw exception("PyObject_CallObject Failed");
}
}
Py_XDECREF(Main);
// c++调用Python的函数 list参数和list返回值
PyObject* TestFun = PyObject_GetAttrString(m, "TestFun");
if (TestFun && PyCallable_Check(TestFun)) {
// 参数准备 参数变量是tuple
PyObject *args = PyTuple_New(1); // 参数是元组 只有元组这个一个参数
// 传递的list对象
PyObject *lis = PyList_New(0);
for (int i = 0; i < 5; i++)
PyList_Append(lis, PyLong_FromLong(i + 100));
// 将list写入元组参数列表中
PyTuple_SetItem(args, 0, lis);
// 函数对象和参数的返回值
PyObject* re = PyObject_CallObject(TestFun, args);
Py_XDECREF(args); // lis也在args中销毁
// 处理返回值
if (re)
{
cout << "PyObject_CallObject return" << endl;
int size = PyList_Size(re);
for (int i = 0; i < size; i++)
{
PyObject *val = PyList_GetItem(re, i);
if (!val)
continue;
printf("[%d]", PyLong_AsLong(val));
}
Py_XDECREF(re);
}
}
Py_XDECREF(TestFun);
結果
可以看到调用成功了,将传递的列表输出出来,返回值也可以输出出来。
三、完整调用代码贴在下面
#include <iostream>
#include <Python.h>
#include <exception>
using namespace std;
int main(int argc, char*argv[])
{
cout << "C++ call Python" << endl;
// 设置Python的Home路径
Py_SetPythonHome(L"./");
// Python初始化解释器
Py_Initialize();
PyObject *m = NULL; // 主模块
try
{
int re = 0;
// 执行Python脚本
re = PyRun_SimpleString("print('Hello world!')");
re = PyRun_SimpleString("print(\"__name__ = \", __name__)");
// 执行Python文件
char* filename = "test.py";
FILE * fp = fopen(filename, "r");
if (!fp)
{
throw exception("open file failed");
}
PyRun_AnyFile(fp, filename);
if (re != 0)
{
PyErr_Print();
throw exception("PyRun_AnyFile failed");
}
// 获取主模块
PyObject *key = PyUnicode_FromString("__main__");
m = PyImport_GetModule(key); // 不清理参数,需要手动清理
Py_XDECREF(key);
// 2-1 调用python的变量 python做配置文件
//con = {
// "width":1920,
// "heigth" : 1080,
// "title" : "C++ call Python"
//}
{
// 根据模块和名称获取对象(对象可以是变量、函数和类)
PyObject* conf = PyObject_GetAttrString(m, "conf");
if (!conf) {
throw exception("conf noe find!");
}
PyObject *key = PyUnicode_FromString("width");
int width = PyLong_AsLong(PyDict_GetItem(conf, key));
Py_XDECREF(key);
key = PyUnicode_FromString("height");
int height = PyLong_AsLong(PyDict_GetItem(conf, key));
Py_XDECREF(key);
key = PyUnicode_FromString("title");
wchar_t title[1024] = { 0 };
int size = PyUnicode_AsWideChar(PyDict_GetItem(conf, key), title, 1023);
Py_XDECREF(key);
printf("width=%d height=%d \n", width, height);
wprintf(L"title=%s\n", title);
Py_XDECREF(conf);
}
{
// 获取类
PyObject* TypePy = PyObject_GetAttrString(m, "TypePy");
if (!TypePy) {
throw exception("TypePy noe find!");
}
// 实例化对象 相当于调用构造函数__init__ 传递构造函数的参数NULL
PyObject *obj = PyObject_CallObject(TypePy, NULL);
if (!obj) {
throw exception("obj not Create!");
}
// 调用类成员函数 i(int) s(string)
PyObject *re = PyObject_CallMethod(obj, "test", "is", 2001, "c Para2");
cout << "PyObject_CallMethod return" << PyLong_AsLong(re) << endl;
Py_XDECREF(re);
// 访问成员变量
PyObject* var = PyObject_GetAttrString(obj, "id");
cout << "TypePy.id=" << PyLong_AsLong(var) << endl;
Py_XDECREF(var);
Py_XDECREF(obj);
Py_XDECREF(TypePy);
}
{
// C++调用Python函数
PyObject* Main = PyObject_GetAttrString(m, "Main");
if (Main && PyCallable_Check(Main)) {
// 函数对象和参数 返回对象
if (!PyObject_CallObject(Main, 0))
{
throw exception("PyObject_CallObject Failed");
}
}
Py_XDECREF(Main);
// c++调用Python的函数 list参数和list返回值
PyObject* TestFun = PyObject_GetAttrString(m, "TestFun");
if (TestFun && PyCallable_Check(TestFun)) {
// 参数准备 参数变量是tuple
PyObject *args = PyTuple_New(1); // 参数是元组 只有元组这个一个参数
// 传递的list对象
PyObject *lis = PyList_New(0);
for (int i = 0; i < 5; i++)
PyList_Append(lis, PyLong_FromLong(i + 100));
// 将list写入元组参数列表中
PyTuple_SetItem(args, 0, lis);
// 函数对象和参数的返回值
PyObject* re = PyObject_CallObject(TestFun, args);
Py_XDECREF(args); // lis也在args中销毁
// 处理返回值
if (re)
{
cout << "PyObject_CallObject return" << endl;
int size = PyList_Size(re);
for (int i = 0; i < size; i++)
{
PyObject *val = PyList_GetItem(re, i);
if (!val)
continue;
printf("[%d]", PyLong_AsLong(val));
}
Py_XDECREF(re);
}
}
Py_XDECREF(TestFun);
}
Py_XDECREF(m);
// 清理python
Py_Finalize();
}
catch (const std::exception&ex)
{
if (PyErr_Occurred())
PyErr_Print();
cout << ex.what() << endl;// 清理python
Py_XDECREF(m);
Py_Finalize();
}
getchar();
return 0;
}
四、总结
- 本文使用C++调用python函数传递list参数并获取返回。
- 如果觉得文章对你有用处,记得
点赞
收藏
转发
一波哦,博主也支持为铁粉丝制作专属动态壁纸哦~
? 往期优质文章分享
- C++ QT结合FFmpeg实战开发视频播放器-01环境的安装和项目部署
- 解决QT问题:运行qmake:Project ERROR: Cannot run compiler ‘cl‘. Output:
- 解决安装QT后MSVC2015 64bit配置无编译器和调试器问题
- Qt中的套件提示no complier set in kit和no debugger,出现黄色感叹号问题解决(MSVC2017)
- Python+selenium 自动化 - 实现自动导入、上传外部文件(不弹出windows窗口)
? 优质教程分享 ?
- ?如果感觉文章看完了不过瘾,可以来我的其他 专栏 看一下哦~
- ?比如以下几个专栏:Python实战微信订餐小程序、Python量化交易实战、C++ QT实战类项目 和 算法学习专栏
- ?可以学习更多的关于C++/Python的相关内容哦!直接点击下面颜色字体就可以跳转啦!
学习路线指引(点击解锁) | 知识定位 | 人群定位 |
---|---|---|
? Python实战微信订餐小程序 ? | 进阶级 | 本课程是python flask+微信小程序的完美结合,从项目搭建到腾讯云部署上线,打造一个全栈订餐系统。 |
?Python量化交易实战 ? | 入门级 | 手把手带你打造一个易扩展、更安全、效率更高的量化交易系统 |
❤️ C++ QT结合FFmpeg实战开发视频播放器❤️ | 难度偏高 | 分享学习QT成品的视频播放器源码,需要有扎实的C++知识! |
? 游戏爱好者九万人社区? | 互助/吹水 | 九万人游戏爱好者社区,聊天互助,白嫖奖品 |
? Python零基础到入门 ? | Python初学者 | 针对没有经过系统学习的小伙伴,核心目的就是让我们能够快速学习Python的知识以达到入门 |
? 资料白嫖,温馨提示 ?
关注下面卡片即刻获取更多编程知识,包括各种语言学习资料,上千套PPT模板和各种游戏源码素材等等资料。更多内容可自行查看哦!
转载请注明:xuhss » Python&C++相互混合调用编程全面实战-17C++调用python函数并传递list参数并获取返回