C++如何調(diào)用簡單的python程序
一、基本環(huán)境的搭建
首先,用vs創(chuàng)建一個(gè)win32的控制臺(tái)應(yīng)用程序項(xiàng)目(vs2017及以上怎么創(chuàng)建百度一下就知道了)。
然后配置好python的環(huán)境變量(把include文件夾加到包含目錄,libs文件夾加到庫目錄,最好還把include文件夾加到附加包含目錄)。
如下圖所示:


最后,把python36.libs文件加到依賴項(xiàng)(如果是debug編譯,要重命名python36.libs為python36_d.libs,然后加到依賴項(xiàng))
如下圖所示。


二、直接在C++里面調(diào)用執(zhí)行python語句
搭建好一中的環(huán)境之后,在源文件里面新建一個(gè)cpp文件,cpp里面的代碼如下。
代碼很簡單,應(yīng)該一看就懂了吧。
#include <Python.h>
#include <stdio.h>
#include <iostream>
using namespace std;
int main()
{
//***python調(diào)用***//
//直接在C++里面執(zhí)行python語句
void usePythonWorld();
usePythonWorld();
system("pause");
}
void usePythonWorld()//調(diào)用無參數(shù)函數(shù)
{
//初始化python模塊:用來分配python解釋器所使用的全局資源
Py_SetPath(L"D:/chengxuanzhuang/anaconda/envs/python3.6/Lib");
Py_Initialize();
if (!Py_IsInitialized())// 檢查初始化是否成功
{
cout << "初始化失敗" << endl;
Py_Finalize();
}
//直接調(diào)用python語句(沒有成功)
PyRun_SimpleString("print('in python'\n)");
Py_Finalize();
}
需要注意的地方是,在使用PyRun_SimpleString之前,一定要使用 Py_Initialize()初始化Python。
三、調(diào)用python腳本文件里面的定義函數(shù)
調(diào)用不含參數(shù)的函數(shù)
搭建好一中的環(huán)境之后,在源文件里面新建一個(gè)cpp文件,cpp里面的代碼如下。
代碼很簡單,應(yīng)該一看就懂了吧。
#include <Python.h>
#include <stdio.h>
#include <iostream>
using namespace std;
int main()
{
//***python調(diào)用***//
//調(diào)用無參數(shù)python函數(shù)
bool InvokeNonParasFuncByAPI(string module, string func);
InvokeNonParasFuncByAPI("PythonGreet", "Hello");
system("pause");
}
bool InvokeNonParasFuncByAPI(string module, string func)
{
Py_SetPath(L"D:/chengxuanzhuang/anaconda/envs/python3.6/Lib");
Py_Initialize();
if (!Py_IsInitialized())
{
return false;
}
try
{
//將當(dāng)前目錄切換到python腳本放置的目錄,因?yàn)槌跏蓟搅藀ython的全局資源下,所以只能只用python語句
PyRun_SimpleString("import sys\n");
PyRun_SimpleString("sys.path.append('./')");//這個(gè)一個(gè)簡單的執(zhí)行python腳本命令的函數(shù),由于路徑是'./',所以python腳本要放在這個(gè)項(xiàng)目的根目錄
//調(diào)用pyhton腳本
PyObject* moduleName = PyUnicode_FromString(module.c_str());
PyObject* pModule = PyImport_Import(moduleName);
if (!pModule)
{
cout << "Import Module Failed" << endl;
return false;
}
PyObject* pFunc = PyObject_GetAttrString(pModule, func.c_str());//調(diào)用腳本里面的函數(shù)
if (!pFunc || !PyCallable_Check(pFunc)) //這里面的這個(gè)函數(shù)是用來判斷方法是否有效
{
cout << "Get Function Failed!" << endl;
return false;
}
PyObject* pResult = PyObject_CallObject(pFunc, nullptr);//使用python腳本里面的函數(shù)
if (!pResult)
{
cout << "Get Result of Function Failed!" << endl;
return false;
}
}
catch (...)
{
PyErr_Print(); //如果出錯(cuò)會(huì)打印出錯(cuò)誤
PyErr_Clear();
Py_Finalize();
return false;
}
Py_Finalize();
return true;
}
調(diào)用含一個(gè)參數(shù)的函數(shù)
搭建好一中的環(huán)境之后,在源文件里面新建一個(gè)cpp文件,cpp里面的代碼如下。
代碼很簡單,應(yīng)該一看就懂了吧。

其中的demo_test是python腳本文件名,注意不是demo_test.py。最后把這個(gè)python腳本文件和cpp文件放在同一個(gè)文件夾下(注意不是放在源文件里,而是找到cpp文件所在的文件夾,放到這個(gè)文件夾里。
如下。如果是在項(xiàng)目里直接創(chuàng)建的python文件,會(huì)自動(dòng)存放到這個(gè)文件夾下)

不出意外,就可以運(yùn)行這個(gè)cpp文件了。
調(diào)用含多個(gè)參數(shù)的函數(shù)
搭建好一中的環(huán)境之后,在源文件里面新建一個(gè)cpp文件,cpp里面的代碼如下。
代碼很簡單,應(yīng)該一看就懂了吧。
#include <Python.h>
#include <stdio.h>
#include <iostream>
#include <vector>
using namespace std;
int main()
{
//***python調(diào)用***//
//調(diào)用含參數(shù)的python函數(shù)
vector<int> paras;
paras.push_back(3);
paras.push_back(4);
bool InvokeFuncbyAPI(string module, string func, vector<int> paras);
InvokeFuncbyAPI("PythonCalc", "Add",paras);
system("pause");
}
bool InvokeFuncbyAPI(string module, string func,vector<int> paras)
{
Py_SetPath(L"D:/chengxuanzhuang/anaconda/envs/python3.6/Lib");
Py_Initialize();
if (!Py_IsInitialized())
{
return false;
}
try
{
//將當(dāng)前目錄切換到python腳本放置的目錄,因?yàn)槌跏蓟搅藀ython的全局資源下,所以只能只用python語句
PyRun_SimpleString("import sys\n");
PyRun_SimpleString("sys.path.append('./')");//這個(gè)一個(gè)簡單的執(zhí)行python腳本命令的函數(shù),由于路徑是'./',所以python腳本要放在這個(gè)項(xiàng)目的根目錄
//調(diào)用pyhton腳本
PyObject* moduleName = PyUnicode_FromString(module.c_str());
PyObject* pModule = PyImport_Import(moduleName);
if (!pModule)
{
cout << "Import Module Failed" << endl;
return false;
}
PyObject* pFunc = PyObject_GetAttrString(pModule, func.c_str());//調(diào)用腳本里面的函數(shù)
if (!pFunc || !PyCallable_Check(pFunc)) //這里面的這個(gè)函數(shù)是用來判斷方法是否有效
{
cout << "Get Function Failed!" << endl;
return false;
}
PyObject* pResult = nullptr; //創(chuàng)建一個(gè)獲取結(jié)果的python數(shù)據(jù)結(jié)構(gòu)
PyObject* args = PyTuple_New((int)paras.size());//建立一個(gè)存放入?yún)⒌膒ython數(shù)據(jù)結(jié)構(gòu)
for (int i = 0; i < (int)paras.size(); i++)
{
PyTuple_SetItem(args, i, PyLong_FromLong(paras[i])); //將C++的參數(shù)轉(zhuǎn)化為python的參數(shù)
}
pResult = PyObject_CallObject(pFunc, args);
if (!pResult)
{
cout << "Get Result of Function Failed!" << endl;
return false;
}
cout << "Result = " << PyLong_AsLong(pResult) << "by python api" << endl;
}
catch (...)
{
PyErr_Print();//如果出錯(cuò)會(huì)打印出錯(cuò)誤
PyErr_Clear();
Py_Finalize();
return false;
}
Py_Finalize();
return true;
}
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
解讀C++編程中派生類的構(gòu)成和創(chuàng)建
這篇文章主要介紹了解讀C++編程中派生類的構(gòu)成和創(chuàng)建,是C++入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-09-09
C++基于easyx圖形庫實(shí)現(xiàn)推箱子游戲
這篇文章主要為大家詳細(xì)介紹了C++基于easyx圖形庫實(shí)現(xiàn)推箱子游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-06-06
Qt(C++)調(diào)用工業(yè)相機(jī)Basler的SDK使用示例
這篇文章主要介紹了Qt(C++)調(diào)用工業(yè)相機(jī)Basler的SDK使用示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03

