python和c++互调

来源:互联网 发布:筹码分布指标源码 编辑:程序博客网 时间:2024/04/29 09:47

1. C/C++调python
    一.关于python库,http://www.python.org/可以从这里下载,编译得到release和debug库。
    二.被调python文件不含有住函数,不能有变量,内存需C/C++来分配。 

test.pydef print_list(list):    print dictdef show():    print "hello world"

test.cpp 代码其它地方拉来#include "Python.h"#include "Python.h"int main(int argc, char** argv){    //初始化Python    //在使用Python系统前,必须使用Py_Initialize对其    //进行初始化。它会载入Python的内建模块并添加系统路    //径到模块搜索路径中。这个函数没有返回值,检查系统    //是否初始化成功需要使用Py_IsInitialized。    Py_Initialize();    // 检查初始化是否成功    if ( !Py_IsInitialized() ) {        return -1;    }    //添加当前路径    //把输入的字符串作为Python代码直接运行,返回0    //表示成功,-1表示有错。大多时候错误都是因为字符串    //中有语法错误。    PyRun_SimpleString("import sys");    PyRun_SimpleString("sys.path.append('./')");    PyObject *pName,*pModule,*pDict,*pFunc,*pArgs;    // 载入名为test的python脚本    pName = PyString_FromString("test");    pModule = PyImport_Import(pName);    if ( !pModule ) {        printf("can't find test.py");        getchar();        return -1;    }    pDict = PyModule_GetDict(pModule);    if ( !pDict ) {        return -1;    }    // 找出函数名为print_list的函数    pFunc = PyDict_GetItemString(pDict, "print_list");    if ( !pFunc || !PyCallable_Check(pFunc) ) {        printf("can't find function [add]");        getchar();        return -1;    }    // 参数进栈    *pArgs;     pArgs = PyTuple_New(2);    // PyObject* Py_BuildValue(char *format, ...)    // 把C++的变量转换成一个Python对象。当需要从    // C++传递变量到Python时,就会使用这个函数。此函数    // 有点类似C的printf,但格式不同。常用的格式有    // s 表示字符串,    // i 表示整型变量,    // f 表示浮点数,    // O 表示一个Python对象。    PyTuple_SetItem(pArgs, 0, Py_BuildValue("i",3));    PyTuple_SetItem(pArgs, 1, Py_BuildValue("i",4));    // 调用Python函数    PyObject_CallObject(pFunc, pArgs);    //下面这段是查找函数foo 并执行foo    pFunc = PyDict_GetItemString(pDict, "foo");    if ( !pFunc || !PyCallable_Check(pFunc) ) {        printf("can't find function [foo]");        getchar();        return -1;    }    pArgs = PyTuple_New(1);    PyTuple_SetItem(pArgs, 0, Py_BuildValue("i",2)); //    PyObject_CallObject(pFunc, pArgs);    Py_DECREF(pName);    Py_DECREF(pArgs);    Py_DECREF(pModule);    // 关闭Python    Py_Finalize();    return 0;}
编译 g++ test.cpp -o cpython -I/usr/include/python2.7 -L/usr/lib/python2.7 -lpython2.7


2. python调C/C++
Python调用C动态链接库
Python调用C库很简单,不经过任何封装打包成so,再使用python的ctypes调用即可。
<test.cpp 生成动态库的源文件>#include <stdio.h>extern "C" {    void display() {        printf("This is Display Function\n");    }}
这里需要注意的是:使用g++编译生成动态库的代码中的函数或者方法时, 需要使用extern "C"来进行编译
g++ test.cpp -fPIC -shared -o libtest.so

<call.py 调用动态库的源文件>import ctypesso = ctypes.CDLL("./libtest.so")  so.display()

Python调用C++(含类,重载)动态链接库
但是调用C++的so就有点麻烦了,网上找了下,大部分都是需要extern "C" 来辅助,也就是说还是只能调用C函数 不能直接调用方法 但是能解析C++方法。
<test.cpp 生成动态库的源文件>#include <Akita/Akita.h>class TestLib{public:    void display();    void display(int a);};void TestLib::display(){    cout<<"First display"<<endl;}void TestLib::display(int a) {    cout<<"Second display"<<endl;}extern "C" {TestLib obj;    void display()    {        obj.display();    }    void display_int()    {        obj.display(2);    }}
g++ test.cpp -fPIC -shared -o libtest.so
使用这种方法有点麻烦 但是可以解决问题。注意到后面还是会有个extern "C" 不然构建后的动态链接库没有这些函数的符号表的。

<call.py 调用动态库的源文件>import ctypesso = ctypes.CDLL("./libtest.so")so.display()so.display_int(1)


运行结果如下:
^[root@:~/Projects/nugget/kvDB-py]#python call.py   
First display
Second display


C/C++调用Python模块
<test.cpp >#include <Akita/Akita.h>#include <Python.h>int main(){    Py_Initialize();    if (!Py_IsInitialized())  return FALSE    PyRun_SimpleString("import sys");    PyRun_SimpleString("sys.path.append('./')");    //import Module    PyObject* pModule = PyImport_ImportModule("hello");    if (!pModule) {        cout<<"Can't import Module!/n"<<endl;        return -1;    }    PyObject* pDict = PyModule_GetDict(pModule);    if (!pDict) {        return -1;    }    //fetch Function    PyObject* pFunHi = PyDict_GetItemString(pDict, "display");    PyObject_CallFunction(pFunHi, "s", "Crazybaby");    Py_DECREF(pFunHi);    //Release    Py_DECREF(pModule);    Py_Finalize();    return 0;}
#g++ test.cpp -I/usr/local/include/python2.7 -ldl -lutil -lpthread -lpython2.7

<call.py>def display(name):    print "hi",name


C++为Python编写扩展模块
Python为C++提供脚本接口。