Python做爲一門腳本解釋語言,自己又很好的結合C++,因此使用Python開發,在性能要求的地方調用C/C++底層庫,這簡直是神器。html
本人的開發環境:Centos 7 + Python 3.5.1 + Qt Creator(只是使用Qt Creator編譯而已,並無使用QT的任何庫)python
將C/C++庫作成和日常同樣的DLL和或者.so,好比:mysql
//.h文件 #include <Python.h> //.cpp文件
//C/C++ my.so 或者my.dll
enter "C" void printHello() { std::cout<<"Hello World"<<std::endl; }
#Python
import ctypes from ctypes import * loadso = ctypes.cdll.LoadLibrary mylib = loadso("./my.so") mylib.printHello()
>>>Hello world
代碼解釋:sql
my.so 有一個C導出函數 printHello()函數
import ctypes : 導入官方的一個庫,顧名思義和C有關性能
loadso = ctypes.cdll.LoadLibrary : loadso 表示加載庫用的函數ui
mylib = loadso("./my.so") //或者loadso("my.dll") 加載my.so庫spa
mylib.printHello() : 調用庫函數指針
上述代碼能正常輸出:Hello World,可是他們沒有互相傳值code
Python和C++互相傳值
//.h文件 #include <Python.h> //.cpp文件
enter "C" int printHello(const char* str) { std::cout<<str<<std::endl; return 1; }
那麼Python的問題就來了
str = create_string_buffer(b"Hello World") #mylib.printHello("Hello World") 這裏死活就是顯示:H,*(str+4)纔是'e',*(str+8) 是'l' 依次類推 print (mylib.printHello(str))
>>>Hello World
>>>1
#因爲對Python不是特別的熟悉 怎麼也作不到顯示C++返回的字符串, Python只能顯示C++返回的字符串子能看到一個地址而已
官方文檔說明:https://docs.python.org/3/extending/extending.html#writing-extensions-in-c
很少說,直接上代碼
//.h文件 原本這是C++鏈接Mysql 我只摘抄部分代#include <Python.h>
//.cpp文件 //傳遞多個參數 Python傳過來的參數在args裏面 PyObject* printfHello(PyObject* self,PyObject* args) { int i=0 const char* str; if (!PyArg_ParseTuple(args, "i|s", &i,&str)) //i 表示整形 s 表示字符串 return PyLong_FromLong(0); print("%d,%s",i,str); return Py_BuildValue("s","OK"); //向Python返回OK字符串 } //映射 知道MFC的一看就懂 static PyMethodDef MyMethods[] = {
{"printfHello", printfHello, METH_VARARGS, //"printHello" 中可調用的函數 METH_VARARGS :帶有參數 METH_NOARGS:無參數
"print"}, //說明
{"connect", connect, METH_VARARGS,
"connect mysql"},
{NULL, NULL, 0, NULL}
};
static PyObject* UtilError;
// 向Python中註冊模塊
static struct PyModuleDef spammodule = {
PyModuleDef_HEAD_INIT,
"libMysqlUtil", //模塊名字 import libMysqlUtil
"C++ Connect Mysql",
-1,
MyMethods
};
//PyInit_libMysqlUtil 注意名字 必定要PyInit_ 加上你的模塊名字 否則Python import 會提示沒有定義 PyInit_你的模塊名字 PyMODINIT_FUNC PyInit_libMysqlUtil(void) { PyObject* m = nullptr; m = PyModule_Create(&spammodule);
//m= Py_InitModule(....) Python 2.7 if(!m) { return m; } UtilError = PyErr_NewException("Util.error",NULL,NULL); Py_INCREF(UtilError); PyModule_AddObject(m,"error",UtilError); return m; }
#python import libMysqlUtil libMysqlUtil.printHello(1,"hello World")
>>>1,hello World
>>>OK
到目前爲止Python 和 C/C++互相通訊,能適應大部分需求,結構體傳值尚未研究,對於類,使用指針就行,C++裏面是指針,在Python中會將指針轉化成整形,Python將這個整形傳給C++的時候使用PyArg_ParseTuple又將整形會變成類指針