zoukankan      html  css  js  c++  java
  • Python调用C++底层库,互相传值问题

      Python作为一门脚本解释语言,本身又很好的结合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)才是'e',*(str+8) 是'l'  依次类推
    print (mylib.printHello(str))
    >>>Hello World
    >>>1
    #由于对Python不是特别的熟悉 怎么也做不到显示C++返回的字符串, Python只能显示C++返回的字符串子能看到一个地址而已

    2.Python扩展C/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又将整形会变成类指针

  • 相关阅读:
    unicodedata.normalize()/使用strip()、rstrip()和lstrip()/encode和decode 笔记(具体可看 《Python Cookbook》3rd Edition 2.9~2.11)
    split与re.split/捕获分组和非捕获分组/startswith和endswith和fnmatch/finditer 笔记
    比较字典推导式/dict()/通过键来构造的字典的速率 笔记
    itertools.groupby()/itertools.compress() 笔记
    operator笔记
    slice.indices()/collections.Counter笔记
    deque/defaultdict/orderedict/collections.namedtuple()/collections.ChainMap() 笔记
    实践中总结出来对heapq的一点理解
    学习笔记(42)- 端到端对话到底是什么
    机器翻译-领域专家
  • 原文地址:https://www.cnblogs.com/beat/p/5567459.html
Copyright © 2011-2022 走看看