本文目錄一覽:
- 1、如何讓python調用C和C++代碼
- 2、python怎樣嵌入c
- 3、怎樣把Python代碼嵌入到C程序
- 4、將python2中的cPickle修改成python3中的pickle的正確用法
- 5、各位老闆,python怎麼把c裡面的結構體讀進來
如何讓python調用C和C++代碼
要搞明白如何讓python調用C/C++代碼(也就是寫python的extension),你需要征服手冊中的Extending embedding厚厚的一章。在昨天花了一個小時看地頭暈腦脹,仍然不知道如何寫python的extension後,查閱了一些其他書籍,最終在Python Programming On Win32書中找到了教程。
1. 首先要明白的是,所謂的python擴展(也就是你提供給python的c/c++代碼,不一定是c/c++代碼,可以是其他語言寫的代碼)是一個dll,並且這個dll放在本機python安裝目錄下的DLLs目錄下(譬如我機器上的路徑是:F:/Program Files/Python25/DLLs),假如我們接下來要寫的擴展module名為mb,python調用的代碼為:import mbmb.showMsg(“Python’s really amazing, I kindda love it!”)
2. 搭建環境,我們要使用python提供的c頭文件和lib庫來進行擴展的開發。
在vs 2005下點擊菜單 “工具”-“選項”, 打開選項對話框,選擇”項目和解決方案-VC++目錄”, 然後在右邊”顯示以下內容的目錄”得comboBox上選擇”包含文件”,添加python的include目錄(我的機器上是”F:/Program Files/Python25/include”),然後選擇庫文件,添加python的libs目錄(我的機器上是”F:/Program Files/Python25/libs”)。
既然擴展是一個dll,接下來我們要建立一個“動態鏈接庫”工程,然後開始寫代碼:
#include python.h //python.h是包含python一些定義的頭文件,在python的include目錄下/*我的python版本是2.5, 因為安裝python後它沒提供debug下的lib庫文件,因此你必須生成release版的dll,
想要生成dll版本的,你要到python官網上自己去下載python源代碼,當然你可以繼續生成release版本的dll,但dll中包含調試信息*/#pragma comment(lib, “python25.lib”)//先不管static PyObject* mb_showMsg(PyObject* self, PyObject *args);/*如果你的擴展是mb,那麼必須實現一個initmb函數,並且從dll中導出這個函數,但我們在python中調用import mb時,python會去dll里去調用
extern “C” __declspec(dllexport) void initmb(){/*當調用mb.showMsg(“Python’s really amazing, I kindda love it!”)時, 相當於你告訴python我有一個showMsg函數,我們怎麼告訴python去調用我們dll里的mb_showMsg函數呢?技巧就是下面的方式,定義一個字典數據結構,key = showMsg, value =mb_showMsg,METH_VARARGS是函數調用方式,仔細查手冊吧*/static PyMethodDef mbMethods[] = {
{“showMsg”, mb_showMsg, METH_VARARGS},
{NULL, NULL, NULL} /*sentinel,哨兵,用來標識結束*/};//告訴python我們的模塊名叫mb, 模塊包含的函數都在mbMethods字典里
PyObject *m = Py_InitModule(“mb”, mbMethods);}/*接下來實現核心功能showMsg*///第一個self參數我們用不着,具體查手冊,第二個參數是python傳給我們的參數,它是一個python的參數tuple
static PyObject* mb_showMsg(PyObject* self, PyObject *args){//我們的showMsg函數需要的是一個字符串參數
const char* msg = NULL;/*調用特殊參數解碼python傳遞給我們的參數,s是string,我們傳遞接收參數的變量地址,
如果你的功能函數需要兩個參數,在PyArg_parseTuple後面繼續添加接受參數的變量地址,
這個函數的原型是類似printf的不定參數的形式
PyAPI_FUNC(int) PyArg_ParseTuple(PyObject *, const char *, …);*/if (!PyArg_ParseTuple(args, “s”, msg))
return NULL;//調用MBint r = ::MessageBox(NULL, “hello”, “Caption:Form C module”, MB_ICONINFORMATION | MB_OK);//返回值return Py_BuildValue(“i”, r);}將上面這段混雜着大量注釋的代碼拷貝到你的編輯器里,然後編譯生成mb.dll,修改後綴成mb.pyd,然後拷貝到python的DLLs目錄下,打開idle(python的交互程序),寫入代碼:import mbmb.showMsg(“Python’s really amazing, I kindda love it!”)
python怎樣嵌入c
用c語言編寫一個動態庫,提供兩個函數,兩個數的整形求和,兩個浮點數的求和。取名為mylib.c。
將c函數文件編譯成so動態庫。運行gcc mylib.c -fPIC -shared -o libtest.so命令,在目錄下可以看到生成的庫文件libtest.so。
Python調用so庫文件。首先導入ctypes,其次用CDLL加載so文件,最後調用對應的函數。將python代碼保存到pydemo.py中。
執行python pydemo.py查看運行結果。
眾多python培訓視頻,盡在python學習網,歡迎在線學習!
怎樣把Python代碼嵌入到C程序
這篇文章主要介紹了將Python代碼嵌入C++程序進行編寫的實例,儘管通常還是Python代碼中調用C++程序的情況較多…需要的朋友可以參考下
把python嵌入的C++裡面需要做一些步驟
安裝python程序,這樣才能使用python的頭文件和庫
在我們寫的源文件中增加“Python.h”頭文件,並且鏈入“python**.lib”庫(還沒搞清楚這個庫時靜態庫還是導出庫,需要搞清楚)
掌握和了解一些python的C語言api,以便在我們的c++程序中使用
常用的一些C API函數
在了解下面的函數之前有必要了解一下**PyObject***指針,python裡面幾乎所有的對象都是使用這個指針來指示的。
Py_Initialize()Py_Finalize()
在調用任何python的c函數之前需要調用的函數,“Py_Initialize”是用來初始化python模塊的,推測是加載初始化加載dll的。對應的在使用python模塊之後用“Py_Finalize”來釋放模塊。
PyImport_ImportModule()
用來載入一個python模塊,這個模塊就是一般的python文件。這裡需要注意的是,在加載這個模塊的時候會執行模塊裡面所有可以執行的語句。包括import導入語句和在函數體之外的所有語句
PyObject_GetAttrString()
返回模塊裡面的函數
Py_BuildValue()
建立一個參數元組,一般都是用這個函數來建立元組,然後將這個元組作為參數傳遞給python裡面的函數。
PyEval_CallObject()
調用函數,並把“Py_BuildValue”建立的元組作為參數傳遞給被調用的函數
源碼實例
下面的實例是在c++代碼中調用Python的函數,傳遞參數並且獲取返回值
test.cpp代碼
[cpp] view plain copy
#include iostream
#include Python.h
using namespace std;
int main(int argc, char* argv[])
{
Py_Initialize(); //初始化
PyObject* pModule = NULL;
PyObject* pFunc = NULL;
PyObject* pParam = NULL;
PyObject* pResult = NULL;
const char* pBuffer = NULL;
int iBufferSize = 0;
pModule = PyImport_ImportModule(“test_python”);
if (!pModule)
{
cout “get module failed!” endl;
exit (0);
}
pFunc = PyObject_GetAttrString(pModule, “main”);
if (!pFunc)
{
cout “get func failed!” endl;
cout int(pFunc) endl;
exit (0);
}
pParam = Py_BuildValue(“(s)”, “HEHEHE”);
pResult = PyEval_CallObject(pFunc,pParam);
if(pResult)
{
if(PyArg_Parse(pResult, “(si)”, pBuffer, iBufferSize))
{
cout pBuffer endl;
cout iBufferSize endl;
}
}
Py_DECREF(pParam);
Py_DECREF(pFunc);
Py_Finalize();
//cout “hello” endl;
return 0;
}
test_python.py代碼
[py] view plain copy
def main(szString):
return (“hello”, 5)
將python2中的cPickle修改成python3中的pickle的正確用法
因為pickle可以把字典、列表等結構化數據存到本地文件,讀取後返回的還是字典、列表等結構化數據。而file.write、file.read存取的對象是字符串。 讀取得到的字符串需要再次加工處理才能轉換為字典、列表等,所以pickle的存在是為了方便特殊類型數據的保存。
報錯:ModuleNotFoundError: No module named ‘cPickle’
原因:python2有cPickle,但是在python3下,是沒有cPickle的;
解決辦法:將cPickle改為pickle即可,但是這樣還是出現編碼的問題
報錯:UnicodeDecodeError: ‘ascii’ codec can’t decode byte 0xe7 in position 0: ordinal not in range(128)錯誤解
原因:python3中的pickle需要指定一下編碼
解決辦法:見下圖
參考文章:
【1】 pickle 在python 2和python 3中兼容性問
【2】 Pickle incompatibility of numpy arrays between Python 2 and 3
各位老闆,python怎麼把c裡面的結構體讀進來
展開全部
閟truct就可以使用結構體了:
import
struct
生成一個結構體實例:
data
=
struct.pack(
‘format_string’,
struct_menber_1,
struct_menber_2,
…
)
其中的format_string用來指定結構體的格式(指明該結構體在C中的定義),由兩部分組成
原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hant/n/244956.html