首頁 >後端開發 >Python教學 >詳解python如何呼叫C/C++底層函式庫與互傳值

詳解python如何呼叫C/C++底層函式庫與互傳值

高洛峰
高洛峰原創
2017-02-20 10:32:431684瀏覽

Python作為一門腳本解釋語言,本身又很好的結合C++,所以使用Python開發,在效能要求的地方呼叫C/C++底層函式庫,這簡直是神器。本文詳細介紹了Python呼叫C/C++底層函式庫,互相傳值問題,下面一起來看看。

前言

開發環境:

Centos 7 + Python 3.5.1 + Qt Creator(只是使用Qt Creator編譯而已,並沒有使用QT的任何庫)

Python調用C/C++庫,我現在能做到的有兩種方式

1.extern “C” 導出(互傳值比較麻煩,不建議使用這種方式):

將C/C++函式庫做成和平常一樣的DLL和或.so,例如:

//.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

程式碼解釋:

my.so 有一個C導出函數printHello()

import ctypes  : 導入官方的一個函式庫,顧名思義和C有關

loadso = ctypes.cdll.LoadLibrary  : loadso 表示載入函式庫用的函式

mylib = loadso(“./my.so”)  //或loadso(“my.dll”)  載入my.so函式庫

mylib.printHello() : 呼叫函式庫函數

上述程式碼能正常輸出:Hello World,但是他們沒有互相傳值

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)才是&#39;e&#39;,*(str+8) 是&#39;l&#39; 依次类推
print (mylib.printHello(str))
>>>Hello World
>>>1
#由于对Python不是特别的熟悉 怎么也做不到显示C++返回的字符串, Python只能显示C++返回的字符串子能看到一个地址而已

2.Python擴充C/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又將整形會變成類別指標。

更多詳解python如何呼叫C/C++底層函式庫與互傳值相關文章請關注PHP中文網!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn