之前的一篇博客提到,喵哥打算用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;
}