前言
开发环境:
Centos 7 + Python 3.5.1 + Qt Creator(只是使用Qt Creator编译而已,并没有使用QT的任何库)
Python调用C/C++库,我现在能做到的有两种方式
1.extern “C” 导出(互相传值比较麻烦,不建议使用这种方式):
将C/C++库做成和平常一样的DLL和或者.so,比如:
1
2
3
4
5
6
7
8
|
//.h文件
#include <Python.h>
//.cpp文件
//C/C++ my.so 或者my.dll
enter "C" void printHello()
{
std::cout<< "Hello World" <<std::endl;
}
|
1
2
3
4
5
6
7
|
#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++互相传值
1
2
3
4
5
6
7
8
|
//.h文件
#include <Python.h>
//.cpp文件
enter "C" int printHello( const char * str)
{
std::cout<<str<<std::endl;
return 1;
}
|
那么Python的问题就来了
1
2
3
4
5
6
|
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++返回的字符串子能看到一个地址而已
|
2.Python扩展C/C++
不多说,直接上代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
//.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; }
|
1
2
3
4
5
|
#python
import libMysqlUtil
libMysqlUtil.printHello( 1 , "hello World" )
>>> 1 ,hello World
>>>OK
|
总结
到目前为止Python和C/C++互相通信,能适应大部分需求,结构体传值还没有研究,对于类,使用指针就行,C++里面是指针,在Python中会将指针转化成整形,Python将这个整形传给C++的时候使用PyArg_ParseTuple又将整形会变成类指针。
好了,以上就是本文的全部内容,希望本文的内容对大家学习python和C/C++能有所帮助。