“探坑”——在C++中执行python脚本的艰辛尝试

之前的一篇博客提到,喵哥打算用C++调用python,然后在python中执行powershell,但是在初次尝试后,喵哥发现这不是一个简单的工作,有很多坑需要注意!此篇文章主要围绕如何在C++中调用python(windows10)来陈述喵哥遇到的一些初学者需要留心的地方。

在VS2013中配置python环境

跟大部分环境配置差不多,主要是在项目里加入头文件和库文件。需要注意的是VS的项目的位数要跟python一致,不然会出现“无法解析的外部符号”。

1.添加头文件(.h)和库文件(.lib)的路径

 在“VC++目录”下的包含目录和库目录分别输入对应python的头文件和库文件路径,喵哥用的是Anaconda管理python,所以不能按照这个找路径。通常,只要在python解释器(python.exe)根目录下找到“include”和“libs”文件夹即可。

2.添加所需要的库文件——.lib和.dll

在上述库文件路径下找到python27.lib,复制全称到上图的附加依赖项中,然后在python解释器的路径下找到python27.dll,复制这个文件到VS解决方案生成的可执行文件路径下(需要生成解决方案才有这个路径)。

3.检查系统环境变量中有无PYTHONHOME变量

检查系统环境变量有无PYTHONHOME变量,并且这个变量对应的值应该是对应python解释器的路径。不然会执行代码出错,虽然可能没有错误显示。不过一般来说,只要是直接安装python的话,都有这个变量的。喵哥用的是Anaconda,所以得自己添加。

4.修改python里的一个头文件

通常需要修改的头文件是pyconfig.h,因为在这个头文件里有这么一行代码:

#	ifdef _DEBUG

#				pragma comment(lib,"python27_d.lib")
#			else
#				pragma comment(lib,"python27.lib")
#			endif /* _DEBUG */

应该是在Debug时,读取python27_d.lib,而在release时,读取python27.lib,python给的只有python27.lib,按理说,只要用release就可以正常执行,但是喵哥遇到的情况是:在release下,还是第一句代码有效。。。所以有谁遇到这样的情况(提示找不到python27_d.lib)不要着急,自己手动修改头文件就好,都改成python27.lib。

另外,喵哥在实际运用中还遇到一些头文件对某些变量重定义的情况,这些基本都是修改头文件,记得标注就好了。

5.实例

python,1.py:

def add(a,b):
    print ("in python function add")
    print ("a = " + str(a))
    print ("b = " + str(b))
    print ("ret = " + str(a+b))
    return

C++:

#include "stdafx.h"
#include "Python.h"

int _tmain(int argc, _TCHAR* argv[])
{
	
	// 初始化Python
	//在使用Python系统前,必须使用Py_Initialize对其
	//进行初始化。它会载入Python的内建模块并添加系统路
	//径到模块搜索路径中。这个函数没有返回值,检查系统
	//是否初始化成功需要使用Py_IsInitialized。
	Py_Initialize();
	// 检查初始化是否成功
	if (!Py_IsInitialized())
	{
		return -1;
	}
	// 添加当前路径
	//把输入的字符串作为Python代码直接运行,返回0
	//表示成功,-1表示有错。大多时候错误都是因为字符串
	//中有语法错误。
	PyRun_SimpleString("import sys");
	PyRun_SimpleString("sys.path.append('./')");
	PyRun_SimpleString("import subprocess");
	PyObject *pName, *pModule, *pDict, *pFunc, *pArgs;
	// 载入名为1的脚本
	pName = PyString_FromString("1");
	pModule = PyImport_Import(pName);
	if (!pModule)
	{
		printf("can't find 1.py");
		getchar();
		return -1;
	}
	pDict = PyModule_GetDict(pModule);
	if (!pDict)
	{
		return -1;
	}
	// 找出函数名为add的函数
	pFunc = PyDict_GetItemString(pDict, "add");
	if (!pFunc || !PyCallable_Check(pFunc))
	{
		printf("can't find function [add]");
		getchar();
		return -1;
	}
	// 参数进栈
	pArgs = PyTuple_New(2);
	//  PyObject* Py_BuildValue(char *format, ...)
	//  把C++的变量转换成一个Python对象。当需要从
	//  C++传递变量到Python时,就会使用这个函数。此函数
	//  有点类似C的printf,但格式不同。常用的格式有
	//  s 表示字符串,
	//  i 表示整型变量,
	//  f 表示浮点数,
	//  O 表示一个Python对象。
	PyTuple_SetItem(pArgs, 0, Py_BuildValue("l", 3));  //"l" (integer) [long int] :将C类型的long转换成Pyhon中的int对象。
	PyTuple_SetItem(pArgs, 1, Py_BuildValue("l", 4));
	// 调用Python函数
	PyObject_CallObject(pFunc, pArgs);
	
	//pFunc = PyDict_GetItemString(pDict, "redis");
	//pArgs = PyTuple_New(1);
	//PyTuple_SetItem(pArgs, 0, Py_BuildValue("")); //
	//PyObject_CallObject(pFunc, pArgs);
	Py_DECREF(pName);
	Py_DECREF(pArgs);
	Py_DECREF(pModule);
	// 关闭Python
	Py_Finalize();

	return 0;
}

 

 

 

 

 

 

 

展开阅读全文

没有更多推荐了,返回首页