C++如何調用簡單的python程序
一、基本環(huán)境的搭建
首先,用vs創(chuàng)建一個win32的控制臺應用程序項目(vs2017及以上怎么創(chuàng)建百度一下就知道了)。
然后配置好python的環(huán)境變量(把include文件夾加到包含目錄,libs文件夾加到庫目錄,最好還把include文件夾加到附加包含目錄)。
如下圖所示:
最后,把python36.libs文件加到依賴項(如果是debug編譯,要重命名python36.libs為python36_d.libs,然后加到依賴項)
如下圖所示。
二、直接在C++里面調用執(zhí)行python語句
搭建好一中的環(huán)境之后,在源文件里面新建一個cpp文件,cpp里面的代碼如下。
代碼很簡單,應該一看就懂了吧。
#include <Python.h> #include <stdio.h> #include <iostream> using namespace std; int main() { //***python調用***// //直接在C++里面執(zhí)行python語句 void usePythonWorld(); usePythonWorld(); system("pause"); } void usePythonWorld()//調用無參數函數 { //初始化python模塊:用來分配python解釋器所使用的全局資源 Py_SetPath(L"D:/chengxuanzhuang/anaconda/envs/python3.6/Lib"); Py_Initialize(); if (!Py_IsInitialized())// 檢查初始化是否成功 { cout << "初始化失敗" << endl; Py_Finalize(); } //直接調用python語句(沒有成功) PyRun_SimpleString("print('in python'\n)"); Py_Finalize(); }
需要注意的地方是,在使用PyRun_SimpleString之前,一定要使用 Py_Initialize()初始化Python。
三、調用python腳本文件里面的定義函數
調用不含參數的函數
搭建好一中的環(huán)境之后,在源文件里面新建一個cpp文件,cpp里面的代碼如下。
代碼很簡單,應該一看就懂了吧。
#include <Python.h> #include <stdio.h> #include <iostream> using namespace std; int main() { //***python調用***// //調用無參數python函數 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 { //將當前目錄切換到python腳本放置的目錄,因為初始化到了python的全局資源下,所以只能只用python語句 PyRun_SimpleString("import sys\n"); PyRun_SimpleString("sys.path.append('./')");//這個一個簡單的執(zhí)行python腳本命令的函數,由于路徑是'./',所以python腳本要放在這個項目的根目錄 //調用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());//調用腳本里面的函數 if (!pFunc || !PyCallable_Check(pFunc)) //這里面的這個函數是用來判斷方法是否有效 { cout << "Get Function Failed!" << endl; return false; } PyObject* pResult = PyObject_CallObject(pFunc, nullptr);//使用python腳本里面的函數 if (!pResult) { cout << "Get Result of Function Failed!" << endl; return false; } } catch (...) { PyErr_Print(); //如果出錯會打印出錯誤 PyErr_Clear(); Py_Finalize(); return false; } Py_Finalize(); return true; }
調用含一個參數的函數
搭建好一中的環(huán)境之后,在源文件里面新建一個cpp文件,cpp里面的代碼如下。
代碼很簡單,應該一看就懂了吧。
其中的demo_test是python腳本文件名,注意不是demo_test.py。最后把這個python腳本文件和cpp文件放在同一個文件夾下(注意不是放在源文件里,而是找到cpp文件所在的文件夾,放到這個文件夾里。
如下。如果是在項目里直接創(chuàng)建的python文件,會自動存放到這個文件夾下)
不出意外,就可以運行這個cpp文件了。
調用含多個參數的函數
搭建好一中的環(huán)境之后,在源文件里面新建一個cpp文件,cpp里面的代碼如下。
代碼很簡單,應該一看就懂了吧。
#include <Python.h> #include <stdio.h> #include <iostream> #include <vector> using namespace std; int main() { //***python調用***// //調用含參數的python函數 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 { //將當前目錄切換到python腳本放置的目錄,因為初始化到了python的全局資源下,所以只能只用python語句 PyRun_SimpleString("import sys\n"); PyRun_SimpleString("sys.path.append('./')");//這個一個簡單的執(zhí)行python腳本命令的函數,由于路徑是'./',所以python腳本要放在這個項目的根目錄 //調用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());//調用腳本里面的函數 if (!pFunc || !PyCallable_Check(pFunc)) //這里面的這個函數是用來判斷方法是否有效 { cout << "Get Function Failed!" << endl; return false; } PyObject* pResult = nullptr; //創(chuàng)建一個獲取結果的python數據結構 PyObject* args = PyTuple_New((int)paras.size());//建立一個存放入參的python數據結構 for (int i = 0; i < (int)paras.size(); i++) { PyTuple_SetItem(args, i, PyLong_FromLong(paras[i])); //將C++的參數轉化為python的參數 } 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();//如果出錯會打印出錯誤 PyErr_Clear(); Py_Finalize(); return false; } Py_Finalize(); return true; }
總結
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Qt(C++)調用工業(yè)相機Basler的SDK使用示例
這篇文章主要介紹了Qt(C++)調用工業(yè)相機Basler的SDK使用示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-03-03