C++調(diào)用python

C++調(diào)用python

在C/C++中嵌入Python,可以使用Python提供的強(qiáng)大功能,通過嵌入Python可以替代動態(tài)鏈接庫形式的接口瞬痘,這樣可以方便地根據(jù)需要修改腳本代碼蒿秦,而不用重新編譯鏈接二進(jìn)制的動態(tài)鏈接庫。至少你可以把它當(dāng)成文本形式的動態(tài)鏈接庫盲再,需要的時(shí)候還可以改一改西设,只要不改變接口, C++的程序一旦編譯好了洲胖,再改就沒那么方便了济榨。

C++調(diào)用Python有兩種方式

第一種方式:通過找到Python模塊,類绿映,方法擒滑,構(gòu)造參數(shù)來調(diào)用。

第二中方式叉弦,就是通過構(gòu)造出一個(gè)Python的腳本丐一,用python引擎來執(zhí)行。

第一種方式可能更為優(yōu)雅淹冰,符合大多數(shù)的反射調(diào)用的特點(diǎn)库车。(如c#的反射機(jī)制,c#調(diào)用Com+樱拴,c#調(diào)用javascript腳本等)柠衍。

一個(gè)問題:兩種語言互相調(diào)用的時(shí)候,需要做數(shù)據(jù)結(jié)構(gòu)(如基本類型晶乔,字符串珍坊,整數(shù)類型等,以及自定義的類等類型)間的轉(zhuǎn)換正罢,共享內(nèi)存中的一個(gè)對象阵漏。比如,如何將C++的對象實(shí)例傳入python中翻具,并在python中使用履怯。c++和python并不在一個(gè)進(jìn)程中,因此可以使用boost的shared_ptr來實(shí)現(xiàn)裆泳。Python調(diào)用C++叹洲,換句話說就是需要把C++封裝成Python可以“理解”的類型。同理可知C++怎么去調(diào)用Python腳本晾虑。

下面這個(gè)例子疹味,主要是演示了c++調(diào)用python仅叫,可以在c++中形成一個(gè)python腳本,然后利用PyRun_SimpleString調(diào)用;并且糙捺,構(gòu)造一個(gè)c++的對象诫咱,傳入到python中,并在python的腳本中調(diào)用其函數(shù)洪灯。

皮皮blog

VS中編譯運(yùn)行

vs安裝配置

安裝python3.4坎缭,然后配置系統(tǒng)環(huán)境變量。

安裝Visual Studio2010(注意可以不用安裝其它好多東西签钩,只要安裝c++就可以了)掏呼。[Visual Studio相關(guān)設(shè)置]

vs中新建一個(gè)win32控制臺應(yīng)用程序,一路確定完成铅檩。

VS2010的配置(設(shè)置編譯環(huán)境)

1. c++調(diào)用python需要在vs2010中的cpp文件中加入憎夷,這個(gè)頭文件在python安裝目錄Python\include下

要成功引入就要把Python.h的頭文件目錄(如D:\python3.4.2\include放在菜單 > 項(xiàng)目 > 屬性 > C/C++ > 常規(guī) > 附加包含目錄下(或者右鍵項(xiàng)目)

2. 還需要一個(gè)python34.lib,如果不導(dǎo)入的話昧旨,會提示你出現(xiàn)這個(gè)文件的缺失拾给。文件在python\libs下,找到此文件之后進(jìn)入VS2010兔沃,菜單 >項(xiàng)目 > 屬性 > 配置屬性 > VC++目錄 > 庫目錄蒋得,把剛才的絕對路徑(如D:\python3.4.2\libs)添加進(jìn)去,此時(shí)變成這樣的了:D:\python3.4.2\libs;$(VCInstallDir)lib;$(VCInstallDir)atlmfc\lib;$(WindowsSdkDir)lib;$(FrameworkSDKDir)\lib乒疏。這樣就可以在VC程序中執(zhí)行python文件了额衙。

或者把D:\python3.4.2\libs放進(jìn)項(xiàng)目 > 屬性 > 配置屬性 > 鏈接器 > 常規(guī) > 附加庫目錄中。

Note: 上面的設(shè)置是對某個(gè)模式生效怕吴,如果將運(yùn)行模式從debug改成了release窍侧,要再進(jìn)去設(shè)置,否則設(shè)置不成功转绷。

程序編輯

編輯c++代碼

將PythonInvoke.cpp文件改成下面的代碼疏之,用于調(diào)用python程序helloworld

// PythonInvoke.cpp : 定義控制臺應(yīng)用程序的入口點(diǎn)。

#include "stdafx.h"

#include

void main(){

Py_Initialize(); /*初始化python解釋器,告訴編譯器要用的python編譯器*/

PyRun_SimpleString("import helloworld"); /*調(diào)用python文件*/

PyRun_SimpleString("helloworld.printHello()");/*調(diào)用python文件中的函數(shù)*/

Py_Finalize(); /*結(jié)束python解釋器暇咆,釋放資源*/

system("pause");

}

Note: 當(dāng)python代碼有錯誤時(shí),PyImport_ImportModule函數(shù)返回NULL丙曙;

另一種調(diào)用方式的代碼

#include//前面所做的一切配置都是為了調(diào)用這個(gè)頭文件和相關(guān)庫

#include

using namespace std;

int main(){

Py_Initialize();//使用python之前爸业,要調(diào)用Py_Initialize();這個(gè)函數(shù)進(jìn)行初始化

PyObject * pModule = NULL;//聲明變量

PyObject * pFunc = NULL;// 聲明變量

pModule =PyImport_ImportModule("helloworld");//這里是要調(diào)用的文件名

pFunc= PyObject_GetAttrString(pModule, "Hello");//這里是要調(diào)用的函數(shù)名

PyEval_CallObject(pFunc, NULL);//調(diào)用函數(shù)

Py_Finalize();//調(diào)用Py_Finalize,這個(gè)根Py_Initialize相對應(yīng)的亏镰。

return 0;

}

編輯python代碼

在項(xiàng)目源文件中扯旷,添加文件命令為helloworld.py

def printHello():

print("Hello World!")

.py文件保存在.cpp同目錄下

Note: 不能將python文件的名字命名為test.py,否則報(bào)錯,由于test.py是python內(nèi)置python腳本文件索抓,也就是python有自己的test.py文件钧忽,并且其優(yōu)先級比你的高毯炮。。

皮皮blog

程序運(yùn)行

如果你安裝的python是64位的耸黑,則vs2010中需要把解決方案平臺定位‘X64’的模式下桃煎,否則配置不成功。

報(bào)錯:fatal error LNK1112: 模塊計(jì)算機(jī)類型“X86”與目標(biāo)計(jì)算機(jī)類型“x64”沖突大刊。[fatal error LNK1112]

Release模式下運(yùn)行

release模式下運(yùn)行不用設(shè)置太多東西

修改運(yùn)行選項(xiàng)

Debug改為Release

win32下拉配置为迈,新建x64,一路確定

注意缺菌,執(zhí)行下面之前要在release模式下再設(shè)置一次VS2010的配置

運(yùn)行

運(yùn)行成功葫辐!

Debug模式下運(yùn)行

1. Debug下,python/libs目錄下的python34.lib需要復(fù)制并重命名為python34_d.lib的形式

設(shè)置:項(xiàng)目 > 屬性 > 配置屬性 > 鏈接器 > 輸入 > 附加依賴庫 > python34_d.lib伴郁。

[源碼編譯 python 生成 python26_d.lib 耿战; 處理 error c101008d]

為什么會有python34.lib和python34_d.lib的差別就是因?yàn)?python_d.lib是 庫的調(diào)試后形式,當(dāng)我們以debug模式編譯工程時(shí)焊傅,python就用這個(gè)lib文件剂陡,但是這個(gè)文件是不可用 的。對于這點(diǎn)租冠,最快的辦法就是強(qiáng)制要求python在任何情況下都是用非調(diào)試版本鹏倘,就可以了。

就是說顽爹,不重命名的解決方法纤泵,對python頭文件python/include/pyconfig.h進(jìn)行修改:

#? ? ? ? ? if defined(_DEBUG)

#? ? ? ? ? ? ? pragma? comment(lib,"python34_d.lib")

#? ? ? ? ? elif defined(Py_LIMITED_API)

#? ? ? ? ? ? ? pragma comment(lib,"python3.lib")

#? ? ? ? ? else

#? ? ? ? ? ? ? pragma comment(lib,"python34.lib")

#? ? ? ? ? endif /* _DEBUG */

將DEBUG條件下的lib由python34_d.lib改為python34.lib.

#? ? ? ? ? if defined(_DEBUG)

#? ? ? ? ? ? ? pragma? comment(lib,"python34.lib")

2. 64位debug下的方案解決

右鍵項(xiàng)目名,點(diǎn)擊屬性镜粤,彈出項(xiàng)目屬性頁捏题,找到鏈接器—高級,修改右側(cè)的目標(biāo)計(jì)算機(jī)肉渴,選擇有X64的那個(gè)選項(xiàng)公荧。如果沒有,則選擇編譯器Configuration Manager中new同规,添加amd64等平臺循狰,然后工程屬性中選擇x64。

這一步好像也不用券勺,只要在運(yùn)行時(shí)選擇x64就可以了绪钥。見3.運(yùn)行。

Note: 屬性 - 鏈接器 - 命令行 -附加選項(xiàng):如果里面有"/MACHINE:I386"之類的,要刪了关炼。

3. 32位庫改成64位庫

項(xiàng)目 > 屬性 > 配置屬性 > Vc++目錄> 庫目錄程腹,這里要將32位庫改成64位庫,相當(dāng)重要儒拂!

$(VCInstallDir)lib\amd64

$(VCInstallDir)atlmfc\lib\amd64

$(WindowsSdkDir)lib\x64

如:將D:\python3.4.2\libs;$(VCInstallDir)lib;$(VCInstallDir)atlmfc\lib;$(WindowsSdkDir)lib;$(FrameworkSDKDir)\lib

換成D:\python3.4.2\libs;$(VCInstallDir)lib\amd64;$(VCInstallDir)atlmfc\lib\amd64;$(WindowsSdkDir)lib\x64;$(FrameworkSDKDir)\lib

沒有這樣設(shè)置會報(bào)錯:msvcprtd.lib(MSVCP100D.dll) : fatal error LNK1112: module machine type ‘X86’ conflicts with target machine type ‘x64’

msvcprtd.lib(MSVCP100D.dll) : fatal error LNK1112: 模塊計(jì)算機(jī)類型“X86”與目標(biāo)計(jì)算機(jī)類型“x64”沖突

[msvcprtd.lib(MSVCP100D.dll) : fatal error LNK1112: module machine type ‘X86’ conflicts with target machine type ‘x64’]

[msvcprtd.lib(MSVCP100D.dll) : fatal error LNK1112: module machine type 'X86' conflicts with target machine type 'x64']

還有一個(gè)cmake類似的問題[fatal error LNK1112: 模塊計(jì)算機(jī)類型“X86”與目標(biāo)計(jì)算機(jī)類型“x64”沖突——我的解決方案]

3. 運(yùn)行:右鍵項(xiàng)目名寸潦,選擇清理解決方案色鸳,清理完成之后重新生成解決方案,然后選擇X64平臺編譯器去debug,便可以調(diào)試成功。

可能存在的錯誤

c++調(diào)用python時(shí)報(bào)錯LINK : fatal error LNK1123: 轉(zhuǎn)換到 COFF 期間失敗: 文件無效或損壞

這個(gè)是由于日志文件引起的鹏氧,可以將項(xiàng)目\屬性\配置屬性\清單工具\(yùn)輸入和輸出\嵌入清單:原來是“是”,改成“否”咏雌。

或者將項(xiàng)目\屬性\配置屬性\鏈接器\清單文件\生成清單:原來是“是”,改成“否”校焦。

如果仍然無效赊抖,判斷是否已經(jīng)安裝了VS2012,如果已經(jīng)安裝寨典,需要安裝VS2010 sp1補(bǔ)丁氛雪。

[鏈接器工具錯誤 LNK1123]

但是如果程序要加寫rc,必須帶清單才能正常使用耸成。所以這種治標(biāo)不治本的方法失效了报亩。

還有一種解決方案:

出現(xiàn)這個(gè)問題的原因:可能是因?yàn)橄到y(tǒng)最近多次更新,出現(xiàn)了兩個(gè)版本的cvtres.exe井氢。而系統(tǒng)變量里將這倆都引用了弦追,編譯的時(shí)候,不知道用哪個(gè)了花竞,導(dǎo)致出錯劲件。所以要刪掉一個(gè)。

一個(gè)在C:\Windows\Microsoft.NET\Framework\v4.0.30319\cvtres.exe约急,另一個(gè)在你安裝VS的軟件目錄..\Microsoft Visual Studio 10.0\vc\bin\cvtres.exe

然后右鍵屬性-->詳細(xì)信息 查看兩者版本號零远,把老的Kill掉,就完了厌蔽。

[徹底解決 LINK : fatal error LNK1123: 轉(zhuǎn)換到 COFF 期間失敗: 文件無效或損壞]

編譯時(shí)_RTC_Shutdown和_RTC_InitBase相關(guān)錯誤的解決方法:

error LNK2001: 無法解析的外部符號 _RTC_Shutdown;error LNK2001: 無法解析的外部符號 _RTC_InitBase

右鍵點(diǎn)擊項(xiàng)目牵辣,修改:屬性 > 配置屬性 > C/C++ > 代碼生成 > 基本運(yùn)行時(shí)檢查,將值從“兩者(......)”改為“默認(rèn)值”奴饮。

[編譯時(shí)_RTC_Shutdown和_RTC_InitBase相關(guān)錯誤的解決方法]

LINK : error LNK2001: 無法解析的外部符號 mainCRTStartup

其它錯誤

error LNK2019: 無法解析的外部符號 __imp_system纬向,該符號在函數(shù) main 中被引用

原因是system("pause");沒有include

[C++調(diào)用python配置及編譯出現(xiàn)的問題]

皮皮blog

命令行中直接調(diào)用執(zhí)行

編譯選項(xiàng), 需要手動指定Python 的include 路徑, 和鏈接接路徑戴卜。

代碼:

g++ Python.cpp -o Python-I/usr/include/python2.5 -L/usr/lib/python2.5-lpython2.5

皮皮blog

C++調(diào)用python帶參數(shù)傳遞

調(diào)用Python函數(shù)時(shí)罢猪,參數(shù)的傳遞,就是c++的類型叉瘩,怎么轉(zhuǎn)換成Python的類型;另外一個(gè)問題是粘捎,Python函數(shù)的返回值薇缅,怎么轉(zhuǎn)換成C++中的類型危彩。

在C程序中用Python腳本傳遞參數(shù),或者獲得Python腳本的返回值泳桦,則要使用更多的函數(shù)來編寫C程序汤徽。由于Python有自己的數(shù)據(jù)類型,因此在C程序中要使用專門的API對相應(yīng)的數(shù)據(jù)類型進(jìn)行操作灸撰。

常用的函數(shù)有以下幾種

1.?dāng)?shù)字與字符串處理

在Python/C?API中提供了Py_BuildValue()函數(shù)對數(shù)字和字符串進(jìn)行轉(zhuǎn)換處理谒府,使之變成Python中相應(yīng)的數(shù)據(jù)類型。其函數(shù)原型如下所示浮毯。

PyObject*?Py_BuildValue(?const?char?*format,?...)

其參數(shù)含義如下完疫。

·?????format:格式化字符串,如表8-1所示债蓝。

Py_BuildValue()函數(shù)中剩余的參數(shù)即要轉(zhuǎn)換的C語言中的整型壳鹤、浮點(diǎn)型或者字符串等。其返回值為PyObject型的指針饰迹。在C語言中芳誓,所有的Python類型都被聲明為PyObject型。

2.列表操作

在Python/C?API中提供了PyList_New()函數(shù)用以創(chuàng)建一個(gè)新的Python列表啊鸭。PyList_New()函數(shù)的返回值為所創(chuàng)建的列表锹淌。其函數(shù)原型如下所示。

PyObject*?PyList_New(?Py_ssize_t?len)

其參數(shù)含義如下赠制。

·?????len:所創(chuàng)建列表的長度赂摆。

當(dāng)列表創(chuàng)建以后,可以使用PyList_SetItem()函數(shù)向列表中添加項(xiàng)憎妙。其函數(shù)原型如下所示库正。

int?PyList_SetItem(?PyObject?*list,?Py_ssize_t?index,?PyObject?*item)

其參數(shù)含義如下。

·?????list:要添加項(xiàng)的列表厘唾。

·?????index:所添加項(xiàng)的位置索引褥符。

·?????item:所添加項(xiàng)的值。

同樣可以使用Python/C?API中PyList_GetItem()函數(shù)來獲取列表中某項(xiàng)的值抚垃。PyList_GetItem()函數(shù)返回項(xiàng)的值喷楣。其函數(shù)原型如下所示。

PyObject*?PyList_GetItem(?PyObject?*list,?Py_ssize_t?index)

其參數(shù)含義如下鹤树。

·?????list:要進(jìn)行操作的列表铣焊。

·?????index:項(xiàng)的位置索引。

Python/C?API中提供了與Python中列表操作相對應(yīng)的函數(shù)罕伯。例如列表的append方法對應(yīng)于PyList_Append()函數(shù)曲伊。列表的sort方法對應(yīng)于PyList_Sort()函數(shù)。列表的reverse方法對應(yīng)于PyList_Reverse()函數(shù)。其函數(shù)原型分別如下所示坟募。

int?PyList_Append(?PyObject?*list,?PyObject?*item)

int?PyList_Sort(?PyObject?*list)

int?PyList_Reverse(?PyObject?*list)

對于PyList_Append()函數(shù)岛蚤,其參數(shù)含義如下。

·?????list:要進(jìn)行操作的列表懈糯。

·?????item:要參加的項(xiàng)涤妒。

對于PyList_Sort()和PyList_Reverse()函數(shù),其參數(shù)含義相同赚哗。

·?????list:要進(jìn)行操作的列表她紫。

3.元組操作

在Python/C?API中提供了PyTuple_New()函數(shù),用以創(chuàng)建一個(gè)新的Python元組屿储。PyTuple_New()函數(shù)返回所創(chuàng)建的元組贿讹。其函數(shù)原型如下所示。

PyObject*?PyTuple_New(?Py_ssize_t?len)

其參數(shù)含義如下扩所。

·?????len:所創(chuàng)建元組的長度围详。

當(dāng)元組創(chuàng)建以后,可以使用PyTuple_SetItem()函數(shù)向元組中添加項(xiàng)祖屏。其函數(shù)原型如下所示助赞。

int?PyTuple_SetItem(?PyObject?*p,?Py_ssize_t?pos,?PyObject?*o)

其參數(shù)含義如下所示。

·?????p:所進(jìn)行操作的元組袁勺。

·?????pos:所添加項(xiàng)的位置索引雹食。

·?????o:所添加的項(xiàng)值。

可以使用Python/C?API中PyTuple_GetItem()函數(shù)來獲取元組中某項(xiàng)的值期丰。PyTuple_GetItem()函數(shù)返回項(xiàng)的值群叶。其函數(shù)原型如下所示。

PyObject*?PyTuple_GetItem(?PyObject?*p,?Py_ssize_t?pos)

其參數(shù)含義如下钝荡。

·?????p:要進(jìn)行操作的元組街立。

·?????pos:項(xiàng)的位置索引。

當(dāng)元組創(chuàng)建以后可以使用_PyTuple_Resize()函數(shù)重新調(diào)整元組的大小埠通。其函數(shù)原型如下所示赎离。

int?_PyTuple_Resize(?PyObject?**p,?Py_ssize_t?newsize)

其參數(shù)含義如下。

·?????p:指向要進(jìn)行操作的元組的指針端辱。

·?????newsize:新元組的大小梁剔。

4.字典操作

在Python/C?API中提供了PyDict_New()函數(shù)用以創(chuàng)建一個(gè)新的字典。PyDict_New()函數(shù)返回所創(chuàng)建的字典舞蔽。其函數(shù)原型如下所示荣病。

PyObject*?PyDict_New()

當(dāng)字典創(chuàng)建后,可以使用PyDict_SetItem()函數(shù)和PyDict_SetItemString()函數(shù)向字典中添加項(xiàng)渗柿。其函數(shù)原型分別如下所示个盆。

int?PyDict_SetItem(?PyObject?*p,?PyObject?*key,?PyObject?*val)

int?PyDict_SetItemString(?PyObject?*p,?const?char?*key,?PyObject?*val)

其參數(shù)含義如下。

·?????p:要進(jìn)行操作的字典。

·?????key:添加項(xiàng)的關(guān)鍵字砾省,對于PyDict_SetItem()函數(shù)其為PyObject型鸡岗,對于PyDict_SetItemString()函數(shù)其為char型。

·?????val:添加項(xiàng)的值编兄。

使用Python/C?API中的PyDict_GetItem()函數(shù)和PyDict_GetItemString()函數(shù)來獲取字典中某項(xiàng)的值。它們都返回項(xiàng)的值声登。其函數(shù)原型分別如下所示狠鸳。

PyObject*?PyDict_GetItem(?PyObject?*p,?PyObject?*key)

PyObject*?PyDict_GetItemString(?PyObject?*p,?const?char?*key)

其參數(shù)含義如下。

·?????p:要進(jìn)行操作的字典悯嗓。

·?????key:添加項(xiàng)的關(guān)鍵字件舵,對于PyDict_GetItem()函數(shù)其為PyObject型,對于PyDict_GetItemString()函數(shù)其為char型脯厨。

使用Python/C?API中的PyDict_DelItem()函數(shù)和PyDict_DelItemString()函數(shù)可以刪除字典中的某一項(xiàng)铅祸。其函數(shù)原型如下所示。

int?PyDict_DelItem(?PyObject?*p,?PyObject?*key)

int?PyDict_DelItemString(?PyObject?*p,?char?*key)

其參數(shù)含義如下合武。

·?????p:要進(jìn)行操作的字典临梗。

·?????key:添加項(xiàng)的關(guān)鍵字,對于PyDict_DelItem()函數(shù)其為PyObject型稼跳,對于PyDict_DelItemString()函數(shù)其為char型盟庞。

使用Python/C?API中的PyDict_Next()函數(shù)可以對字典進(jìn)行遍歷。其函數(shù)原型如下所示汤善。

int?PyDict_Next(?PyObject?*p,?Py_ssize_t?*ppos,?PyObject?**pkey,?PyObject?**pvalue)

其參數(shù)含義如下什猖。

·?????p:要進(jìn)行遍歷的字典。

·?????ppos:字典中項(xiàng)的位置红淡,應(yīng)該被初始化為0不狮。

·?????pkey:返回字典的關(guān)鍵字。

·?????pvalue:返回字典的值在旱。

在Python/C?API中提供了與Python中字典操作相對應(yīng)的函數(shù)摇零。例如字典的item方法對應(yīng)于PyDict_Items()函數(shù)。字典的keys方法對應(yīng)于PyDict_Keys()函數(shù)颈渊。字典的values方法對應(yīng)于PyDict_Values()函數(shù)遂黍。其函數(shù)原型分別如下所示。

PyObject*?PyDict_Items(?PyObject?*p)

PyObject*?PyDict_Keys(?PyObject?*p)

PyObject*?PyDict_Values(?PyObject?*p)

其參數(shù)含義如下俊嗽。

·?????p:要進(jìn)行操作的字典雾家。

5.釋放資源

Python使用引用計(jì)數(shù)機(jī)制對內(nèi)存進(jìn)行管理,實(shí)現(xiàn)自動垃圾回收绍豁。在C/C++中使用Python對象時(shí)芯咧,應(yīng)正確地處理引用計(jì)數(shù),否則容易導(dǎo)致內(nèi)存泄漏。在Python/C?API中提供了Py_CLEAR()敬飒、Py_DECREF()等宏來對引用計(jì)數(shù)進(jìn)行操作邪铲。

當(dāng)使用Python/C?API中的函數(shù)創(chuàng)建列表、元組无拗、字典等后带到,就在內(nèi)存中生成了這些對象的引用計(jì)數(shù)。在對其完成操作后應(yīng)該使用Py_CLEAR()英染、Py_DECREF()等宏來銷毀這些對象揽惹。其原型分別如下所示。

void?Py_CLEAR(?PyObject?*o)

void?Py_DECREF(?PyObject?*o)

其參數(shù)含義如下四康。

·?????o:要進(jìn)行操作的對象搪搏。

對于Py_CLEAR()其參數(shù)可以為NULL指針,此時(shí)闪金,Py_CLEAR()不進(jìn)行任何操作疯溺。而對于Py_DECREF()其參數(shù)不能為NULL指針,否則將導(dǎo)致錯誤哎垦。

6.模塊與函數(shù)

使用Python/C?API中的PyImport_Import()函數(shù)可以在C程序中導(dǎo)入Python模塊囱嫩。PyImport_Import()函數(shù)返回一個(gè)模塊對象。其函數(shù)原型如下所示撼泛。

PyObject*?PyImport_Import(?PyObject?*name)

其參數(shù)含義如下挠说。

·?????name:要導(dǎo)入的模塊名。

使用Python/C?API中的PyObject_CallObject()函數(shù)和PyObject_CallFunction()函數(shù)愿题,可以在C程序中調(diào)用Python中的函數(shù)损俭。其參數(shù)原型分別如下所示。

PyObject*?PyObject_CallObject(?PyObject?*callable_object,?PyObject?*args)

PyObject*?PyObject_CallFunction(?PyObject?*callable,?char?*format,?...)

對于PyObject_CallObject()函數(shù)潘酗,其參數(shù)含義如下杆兵。

·?????callable_object:要調(diào)用的函數(shù)對象。

·?????args:元組形式的參數(shù)列表仔夺。

對于PyObject_CallFunction()函數(shù)琐脏,其參數(shù)含義如下。

·?????callable_object:要調(diào)用的函數(shù)對象缸兔。

·?????format:指定參數(shù)的類型日裙。

·?????...:向函數(shù)傳遞的參數(shù)。

使用Python/C?API中的PyModule_GetDict()函數(shù)可以獲得Python模塊中的函數(shù)列表惰蜜。PyModule_GetDict()函數(shù)返回一個(gè)字典昂拂。字典中的關(guān)鍵字為函數(shù)名,值為函數(shù)的調(diào)用地址抛猖。其函數(shù)原型如下所示格侯。

PyObject*?PyModule_GetDict(?PyObject?*module)

其參數(shù)含義如下鼻听。

·?????module:已導(dǎo)入的模塊對象。

8.2.3?在C中嵌入Python實(shí)例

在VC++?6.0中新建一個(gè)名為“EmbPython”的空“Win32?Console?Application”工程联四。向其添加如下所示的“EmbPython.c”文件撑碴。

程序輸出如下所示。

-==在C中嵌入Python==-

使用Python中的sum函數(shù)求解下列數(shù)之和

0???????1???????2???????3???????4

Using?Function?sum

The?result?is:?10

使用Python中的函數(shù)分割以下字符串:

this?is?an?example

結(jié)果如下所示:

this

is

an

example

按回車鍵退出程序

[Python嵌入C/C++ (Python核心編程)]

C++轉(zhuǎn)換成Python類型朝墩,Py_BuildValue()

http://www.python.org/doc/1.5.2p2/ext/buildValue.html

PyObject*?pArgs=PyTuple_New(1);?//有幾個(gè)參數(shù)醉拓,就是幾

PyTuple_SetItem(pArgs,0,Py_BuildValue("i",3));??//初始第一個(gè)參數(shù),數(shù)據(jù)類型是i收苏,就是int廉嚼,值是3

返回值轉(zhuǎn)換如,PyArg_ParseTuple[PyArg_ParseTuple]

皮皮blog

python調(diào)用c++問題

“如果沒有參數(shù)從python到C++, 是正常的倒戏,但是有參數(shù)就廢了報(bào)錯”。我嘗試實(shí)現(xiàn)并找到答案恐似。

http://stackoverflow.com/questions/145270/calling-c-c-from-python

1. 無參數(shù) 函數(shù)聲明C可用函數(shù)

2. 有參數(shù) 那么實(shí)用SWIG? 也就是我們需要一個(gè)接口文件 即

z.i file

%{

#include "z.h"

extern 函數(shù)名(參數(shù)1杜跷, 參數(shù)2,...);

%}

SWIG在不同語言互相調(diào)用發(fā)揮很重要的作用矫夷。

相關(guān)話題

SWIG

有一個(gè)外部工具叫SWIG葛闷,是Simplified Wrapper and Interface Generator 的縮寫。其作者為David Beazley双藕,同時(shí)也是Python Essential Referenc 一書的作者淑趾。這個(gè)工具可以根據(jù)特別注釋過的C/C++頭文件生成能給Python,Tcl 和Perl 使用的包裝代碼忧陪。使用SWIG 可以省去你寫前面所說的樣板代碼的時(shí)間扣泊。你只要關(guān)心怎么用C/C++解決你的實(shí)際問題就好了。你所要做的就是按SWIG 的格式編寫文件嘶摊,其余的就都由SWIG 來完成延蟹。你可以通過下面的網(wǎng)址找到關(guān)于SWIG 的更多信息。

http://swig.org

Pyrex

創(chuàng)建C/C++擴(kuò)展的一個(gè)很明顯的壞處是你必須要寫C/C++代碼叶堆。你能利用它們的優(yōu)點(diǎn)阱飘,但更重要的是,你也會碰到它們的缺點(diǎn)虱颗。Pyrex 可以讓你只取擴(kuò)展的優(yōu)點(diǎn)沥匈,而完全沒有后顧之憂。它是一種更偏向Python 的C 語言和Python 語言的混合語言忘渔。事實(shí)上高帖,Pyrex 的官方網(wǎng)站上就說“Pyrex 是具有C 數(shù)據(jù)類型的Python“。你只要用Pyrex 的語法寫代碼辨萍,然后運(yùn)行Pyrex 編譯器去編譯源代碼棋恼。Pyrex會生成相應(yīng)的C 代碼返弹,這些代碼可以被編譯成普通的擴(kuò)展。你可以在它的官方網(wǎng)站下載到Pyrex:

http://cosc.canterbury.ac.nz/~greg/python/Pyrex

Psyco

Pyrex 免去了我們再去寫純C 代碼的麻煩爪飘。不過义起,你要去學(xué)會它的那一套與眾不同的語法。最后师崎,你的Pyrex 代碼還是會被轉(zhuǎn)成C 的代碼默终。無論你用C/C++,C/C++加上SWIG犁罩,還是Pyrex齐蔽,都是因?yàn)槟阆胍涌炷愕某绦虻乃俣取H绻憧梢栽诓桓膭幽愕腜ython 代碼的同時(shí)床估,又能獲得速度的提升含滴,那該多好啊。

Psyco 的理念與其它的方法截然不同丐巫。與其改成C 的代碼谈况,為何不讓你已有的Python 代碼

運(yùn)行的更快一些呢?Psyco 是一個(gè)just-in-time(JIT)編譯器递胧,它能在運(yùn)行時(shí)自動把字節(jié)碼轉(zhuǎn)為本地代碼運(yùn)行碑韵。所以,

你只要(在運(yùn)行時(shí))導(dǎo)入Psyco 模塊缎脾,然后告訴它要開始優(yōu)化代碼就可以了祝闻。而不用修改自己的代

碼。Psyco 也可以檢查你代碼各個(gè)部分的運(yùn)行時(shí)間遗菠,以找出瓶頸所在联喘。你甚至可以打開日志功能,來

查看Psyco 在優(yōu)化你的代碼的時(shí)候舷蒲,都做了些什么耸袜。你可以訪問以下網(wǎng)站獲取更多的信息:

http://psyco.sf.net

嵌入

嵌入是Python 的另一功能。與把C 代碼包裝到Python 中的擴(kuò)展相對的牲平,嵌入是把Python 解釋器包裝到C 的程序中堤框。這樣做可以給大型的,單一的纵柿,要求嚴(yán)格的蜈抓,私有的并且(或者)極其重要的應(yīng)用程序內(nèi)嵌Python 解釋器的能力。一旦內(nèi)嵌了Python昂儒,世界完全不一樣了沟使。

Python 提供了很多官方文檔供寫擴(kuò)展的人參考:

擴(kuò)展與嵌入

http://docs.python.org/ext

Python/C API

http://docs.python.org/api

分發(fā)Python 模塊

http://docs.python.org/dist

[python核心編程2e.d. - 擴(kuò)展Python]

from:http://blog.csdn.net/pipisorry/article/details/49532341

ref:python doc:Python/C API Reference Manual

python doc:EmbeddingPython in Another Application

C++中調(diào)用Python腳本

C++調(diào)用Python

C++調(diào)Python示例

在VS2012中C++嵌入python

[如何實(shí)現(xiàn) C/C++ 與 Python 的通信?]*

[Google開發(fā)了一個(gè)框架讓C++和Python交互更容易]讓開發(fā)者為C++程序生成Python的接口包裝器

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末渊跋,一起剝皮案震驚了整個(gè)濱河市腊嗡,隨后出現(xiàn)的幾起案子着倾,更是在濱河造成了極大的恐慌,老刑警劉巖燕少,帶你破解...
    沈念sama閱讀 206,839評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件卡者,死亡現(xiàn)場離奇詭異,居然都是意外死亡客们,警方通過查閱死者的電腦和手機(jī)崇决,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來底挫,“玉大人恒傻,你說我怎么就攤上這事〗ǖ耍” “怎么了盈厘?”我有些...
    開封第一講書人閱讀 153,116評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長官边。 經(jīng)常有香客問我扑庞,道長,這世上最難降的妖魔是什么拒逮? 我笑而不...
    開封第一講書人閱讀 55,371評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮臀规,結(jié)果婚禮上滩援,老公的妹妹穿的比我還像新娘。我一直安慰自己塔嬉,他們只是感情好玩徊,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著谨究,像睡著了一般恩袱。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上胶哲,一...
    開封第一講書人閱讀 49,111評論 1 285
  • 那天畔塔,我揣著相機(jī)與錄音,去河邊找鬼鸯屿。 笑死澈吨,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的寄摆。 我是一名探鬼主播谅辣,決...
    沈念sama閱讀 38,416評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼婶恼!你這毒婦竟也來了桑阶?” 一聲冷哼從身側(cè)響起柏副,我...
    開封第一講書人閱讀 37,053評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎蚣录,沒想到半個(gè)月后割择,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,558評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡包归,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評論 2 325
  • 正文 我和宋清朗相戀三年锨推,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片公壤。...
    茶點(diǎn)故事閱讀 38,117評論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡换可,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出厦幅,到底是詐尸還是另有隱情沾鳄,我是刑警寧澤,帶...
    沈念sama閱讀 33,756評論 4 324
  • 正文 年R本政府宣布确憨,位于F島的核電站译荞,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏休弃。R本人自食惡果不足惜吞歼,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望塔猾。 院中可真熱鬧篙骡,春花似錦、人聲如沸丈甸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,315評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽睦擂。三九已至得湘,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間顿仇,已是汗流浹背淘正。 一陣腳步聲響...
    開封第一講書人閱讀 31,539評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留臼闻,地道東北人跪帝。 一個(gè)月前我還...
    沈念sama閱讀 45,578評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像些阅,于是被迫代替她去往敵國和親伞剑。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評論 2 345

推薦閱讀更多精彩內(nèi)容