C++如何調(diào)用簡單的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++里面調(diào)用執(zhí)行python語句
搭建好一中的環(huán)境之后,在源文件里面新建一個cpp文件,cpp里面的代碼如下。
代碼很簡單,應該一看就懂了吧。
#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)境之后,在源文件里面新建一個cpp文件,cpp里面的代碼如下。
代碼很簡單,應該一看就懂了吧。
#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
{
//將當前目錄切換到python腳本放置的目錄,因為初始化到了python的全局資源下,所以只能只用python語句
PyRun_SimpleString("import sys\n");
PyRun_SimpleString("sys.path.append('./')");//這個一個簡單的執(zhí)行python腳本命令的函數(shù),由于路徑是'./',所以python腳本要放在這個項目的根目錄
//調(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)) //這里面的這個函數(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(); //如果出錯會打印出錯誤
PyErr_Clear();
Py_Finalize();
return false;
}
Py_Finalize();
return true;
}
調(diào)用含一個參數(shù)的函數(shù)
搭建好一中的環(huán)境之后,在源文件里面新建一個cpp文件,cpp里面的代碼如下。
代碼很簡單,應該一看就懂了吧。

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

不出意外,就可以運行這個cpp文件了。
調(diào)用含多個參數(shù)的函數(shù)
搭建好一中的環(huán)境之后,在源文件里面新建一個cpp文件,cpp里面的代碼如下。
代碼很簡單,應該一看就懂了吧。
#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
{
//將當前目錄切換到python腳本放置的目錄,因為初始化到了python的全局資源下,所以只能只用python語句
PyRun_SimpleString("import sys\n");
PyRun_SimpleString("sys.path.append('./')");//這個一個簡單的執(zhí)行python腳本命令的函數(shù),由于路徑是'./',所以python腳本要放在這個項目的根目錄
//調(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)) //這里面的這個函數(shù)是用來判斷方法是否有效
{
cout << "Get Function Failed!" << endl;
return false;
}
PyObject* pResult = nullptr; //創(chuàng)建一個獲取結(jié)果的python數(shù)據(jù)結(jié)構(gòu)
PyObject* args = PyTuple_New((int)paras.size());//建立一個存放入?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();//如果出錯會打印出錯誤
PyErr_Clear();
Py_Finalize();
return false;
}
Py_Finalize();
return true;
}
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
關(guān)于在MFC中將窗口最小化到托盤實現(xiàn)原理及操作步驟
最小化的原理:首先要將窗口隱藏,然后在右下角繪制圖標;恢復的原理:將窗口顯示,再將托盤中的圖片刪除,接下來介紹實現(xiàn)方法,感興趣的朋友可以了解下啊,希望本文對你有所幫助2013-01-01
C++?opencv實現(xiàn)在圖片上畫一條線示例代碼
這篇文章主要為大家介紹了C++?opencv實現(xiàn)在圖片上畫一條線的示例代碼,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-05-05
OpenCV實現(xiàn)圖像轉(zhuǎn)換為漫畫效果
這篇文章主要為大家詳細介紹了OpenCV實現(xiàn)圖像轉(zhuǎn)換為漫畫效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-08-08
Qt串口通信開發(fā)之QSerialPort模塊簡單使用方法與實例
這篇文章主要介紹了Qt串口通信開發(fā)之QSerialPort模塊簡單使用方法與實例,需要的朋友可以參考下2020-03-03

