Python&C++相互混合调用编程全面实战-17C++调用python函数并传递list参数并获取返回

作者:虚坏叔叔
博客:https://xuhss.com

早餐店不会开到晚上,想吃的人早就来了!😄

C++调用python函数传递list参数并获取返回

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-WMU9aIxw-1655596792059)(02.assets/1-1655596705692.png)]

一、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参数并获取返回。
  • 如果觉得文章对你有用处,记得 点赞 收藏 转发 一波哦,博主也支持为铁粉丝制作专属动态壁纸哦~

💬 往期优质文章分享

🚀 优质教程分享 🚀

  • 🎄如果感觉文章看完了不过瘾,可以来我的其他 专栏 看一下哦~
  • 🎄比如以下几个专栏:Python实战微信订餐小程序、Python量化交易实战、C++ QT实战类项目 和 算法学习专栏
  • 🎄可以学习更多的关于C++/Python的相关内容哦!直接点击下面颜色字体就可以跳转啦!
学习路线指引(点击解锁)知识定位人群定位
🧡 Python实战微信订餐小程序 🧡进阶级本课程是python flask+微信小程序的完美结合,从项目搭建到腾讯云部署上线,打造一个全栈订餐系统。
💛Python量化交易实战 💛入门级手把手带你打造一个易扩展、更安全、效率更高的量化交易系统
❤️ C++ QT结合FFmpeg实战开发视频播放器❤️难度偏高分享学习QT成品的视频播放器源码,需要有扎实的C++知识!
💚 游戏爱好者九万人社区💚互助/吹水九万人游戏爱好者社区,聊天互助,白嫖奖品
💙 Python零基础到入门 💙Python初学者针对没有经过系统学习的小伙伴,核心目的就是让我们能够快速学习Python的知识以达到入门

🚀 资料白嫖,温馨提示 🚀

关注下面卡片即刻获取更多编程知识,包括各种语言学习资料,上千套PPT模板和各种游戏源码素材等等资料。更多内容可自行查看哦!

请添加图片描述

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

虚坏叔叔

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值