c++调用python

环境:VS2017,anaconda中python3.7.7
配置结果:debug x64

环境配置

包含目录D:\anaconda3\include
库目录:D:\anaconda3\libs
依赖项:python37_d.lib
1.项目右击-》属性-》c/c+±》附加包含目录
在这里插入图片描述
2.项目右击-》属性-》链接器-》常规-》附加库目录
在这里插入图片描述
2.项目右击-》属性-》链接器-》输入-》附加依赖项
在这里插入图片描述

代码

使用c++调用python代码存在两种方法

1.使用PyRun_SimpleString命令,(这种形式最简单,好用,但是变量不好传递,只能够传递文件名,利于实时性要求不高)

#include <Python.h>
#include<iostream>

using namespace std;

int main()
{
	Py_SetPythonHome(L"D:/anaconda3");//这里地址一定要写对啊!
	/**	添加python.exe所在路径,不添加在运行时出现
	Fatal Python error: initfsencoding: unable to load the file system codec
	ModuleNotFoundError: No module named 'encodings'**/

	Py_Initialize();//使用python之前,要调用Py_Initialize();这个函数进行初始化
	if (!Py_IsInitialized())
	{
		printf("初始化失败!");
		return 0;
	}
	else {
		PyRun_SimpleString("import sys");
		PyRun_SimpleString("sys.path.append('./')");//这一步很重要,修改Python路径
		PyRun_SimpleString("import Sample");

		char *temp = "print(Sample.hello('52,589'))";

		PyRun_SimpleString(temp);
	}
Py_Finalize();//调用Py_Finalize,这个根Py_Initialize相对应的。
		
	return 0;
}

在这里插入图片描述

2.使用PyImport_ImportModule,PyObject_GetAttrString,Py_BuildValue模型传递命令,(这种形式有点繁琐,但是变量传递简单,利于实时性要求高)

#include <Python.h>
#include<iostream>
#include <string>
using namespace std;

int main()
{
	Py_SetPythonHome(L"D:/anaconda3");//这里地址一定要写对啊!
//	/**	添加python.exe所在路径,不添加在运行时出现
//	Fatal Python error: initfsencoding: unable to load the file system codec
//	ModuleNotFoundError: No module named 'encodings'**/itialize();//使用python之前,要调用Py_Initialize();这个函数进行初始化
	if (!Py_IsInitialized())
	{
		printf("初始化失败!");
		return 0;
	}
	else {
		PyRun_SimpleString("import sys");
		PyRun_SimpleString("sys.path.append('./')");//这一步很重要,修改Python路径
		PyObject * pModule = NULL;//声明变量
		PyObject * pFunc = NULL;// 声明变量
		PyRun_SimpleString("import Sample");



		pModule = PyImport_ImportModule("Sample");//这里是要调用的文件名hello.py
		if (pModule == NULL)
		{
			cout << "没找到该Python文件" << endl;
		}
		else {

			//进行整数传参,返回值
			pFunc = PyObject_GetAttrString(pModule, "add");//这里是要调用的函数名
			PyObject* args = Py_BuildValue("(ii)", 28, 103);//给python函数参数赋值
			PyObject* pRet = PyObject_CallObject(pFunc, args);//调用函数
			int res = 0;
			PyArg_Parse(pRet, "i", &res);//转换返回类型
			cout << "整数返回值:" << res << endl;//输出结果

			//进行字符传参,返回值
			pFunc = PyObject_GetAttrString(pModule, "hello");//这里是要调用的函数名
			args = Py_BuildValue("ss", "28", "103");//给python函数参数赋值
			pRet = PyObject_CallObject(pFunc, args);//调用函数
			char* res2;
			PyArg_Parse(pRet, "s", &res2);//转换返回类型
			cout << "字符串返回值:" << res2 << endl;//输出结果
			//printf("%s\n", res);

		}
		Py_Finalize();//调用Py_Finalize,这个根Py_Initialize相对应的。
	}
	return 0;
}

在这里插入图片描述

对应python代码

import numpy as np



def hello(x='12',y='1'):
    if not isinstance(x, str):
        x="helloxx! {}".format(x)
    else:
        x=("helloxx!"+x)
    return x
 
def add(a, b):
    return a+b


# def

if __name__ == '__main__':
    xx = np.array([1, 23, 4156])
    print(xx)
    zz=hello('52889898')
    print(zz)

错误

1.Fatal Python error: initfsencoding: unable to load the file system codec
ModuleNotFoundError: No module named ‘encodings’

Current thread 0x000054b8 (most recent call first):
在这里插入图片描述
解决方法:加上Py_SetPythonHome(L"D:/anaconda3");
在这里插入图片描述

传递参数类型

在这里插入图片描述
参考1:2Vs2017调用Python函数,2VS2017
参考2:这里要隆重的写一下C++调用python遇到的各种坑,搞了好久,终于成功了!

### 如何从C++调用Python代码 要实现从C++调用Python代码的功能,通常会利用Python的嵌入功能。通过这种方式,可以在C++程序中初始化Python解释器并执行Python脚本或函数。以下是具体方法以及相关细节: #### 初始化Python解释器 在C++调用Python的第一步是初始化Python解释器。这可以通过`Py_Initialize()`函数完成[^2]。此函数启动Python运行环境,并允许后续加载模块和执行脚本。 ```cpp #include <Python.h> int main() { Py_Initialize(); // 启动Python解释器 } ``` #### 执行简单的Python语句 一旦Python解释器被成功初始化,就可以使用`PyRun_SimpleString()`来执行单条或多条Python命令。这种方法适用于快速测试或者简单操作场景。 ```cpp PyRun_SimpleString("print('Hello from Python!')"); ``` 上述代码将在控制台打印消息“Hello from Python!”。 #### 调用已定义的Python函数 如果目标是从C++调用特定的Python函数,则需要更复杂的处理流程。首先,在C++端导入包含所需函数的Python模块;其次定位该函数对象;最后准备参数并通过适当接口触发它。 假设有一个名为 `example.py` 的文件,其中定义了一个接受两个整数作为输入并返回其乘积的函数 `multiply(a, b)`: ```python def multiply(a, b): return a * b ``` 对应的C++代码如下所示: ```cpp PyObject* pName = PyUnicode_DecodeFSDefault("example"); PyObject* pModule = PyImport_Import(pName); // 导入module 'example' if (!pModule) { /* 错误处理 */ } // 获取function object 'multiply' PyObject* pFunc = PyObject_GetAttrString(pModule,"multiply"); if(!pFunc || !PyCallable_Check(pFunc)) {/*错误处理*/} // 创建arguments tuple (for two integers) PyObject* argsTuple = PyTuple_Pack(2, PyLong_FromLong(arg1), PyLong_FromLong(arg2)); // Call function & get result as PyObject* PyObject* pValue = PyObject_CallObject(pFunc,argsTuple); long returnValue; if(PyLong_Check(pValue)) { returnValue=PyLong_AsLong(pValue); } /* 清理资源 */ Py_DECREF(argsTuple); Py_XDECREF(pFunc); Py_DECREF(pModule); Py_DECREF(pName); return returnValue; ``` 以上片段展示了如何安全地获取外部Python模块中的函数指针、构造必要的参数列表、实际调用函数并将结果转换回本地数据类型。 #### 结束Python解释器 当不再需要继续使用Python时,应该关闭解释器以释放所有关联资源。这是通过调用`Py_FinalizeEx()`完成的。 ```cpp Py_FinalizeEx(); ``` 注意:虽然这里只讨论了基本概念和技术要点,但在真实项目里还需要考虑异常捕获机制以及其他边界情况管理等问题。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值