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-09C++基于easyx圖形庫實(shí)現(xiàn)推箱子游戲
這篇文章主要為大家詳細(xì)介紹了C++基于easyx圖形庫實(shí)現(xiàn)推箱子游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-06-06Qt(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