ubuntu下将c++转换为so文件并利用python调用

python调用.so

LINUX下C++生成.so文件及编译生成可执行文件的过程

[linux][原创]C++ so库的编译python调用

python 打包成 so | python 调用cpp | python 调用C++简单例子

C++转so文件

这里以简单的加法为例,代码如下,add传入两个int的参数,完成加法。主函数main调用add,return结果。这里需要注意的点在于,不能忘记将函数添加到extern C中。原因是python 的ctype可以调用C而无法调用c++,加上extern "C"后,会指示编译器这部分代码按C语言(而不是C++)的方式进行编译。

#include <iostream>
using namespace std;

extern "C"{

   double add(int, int);

}
double add(int x1, int x2)
{
    return x1+x2;
}
int main()
{
  int a = 1;
  int b =2 ;
  int c;
  c = add(a,b);
  return c;
}

之后打开窗口,输入以下命令,即将上述cpp文件转换为了so文件,注意so文件的名称必须以lib开头。

g++ add.cpp -fpic -shared -o libadd.so

python调用so文件

import ctypes  
ll = ctypes.cdll.LoadLibrary   

lib = ll("./libadd.so") 
input1 = 100
input2 = 220
result1 = lib.add(input1,input2)
result2 = lib.main()
print(result1,result2)
print '***finish***' 

python调用.so多个C++文件转so文件可以看到最后结果为(320,3)

如果c++中add函数的输入为double,则python调用so文件时需要将输入input1 input2,以及add的输出都转换为double的形式,代码如下:

lib.add.restype = ctypes.c_double
result1 = lib.add(ctypes.c_double(input1),ctypes.c_double(input2))

多个C++文件转so文件

下面将多个cpp文件转换为so文件,下面是一个例子:

其中convert.cpp是需要调用的函数,如下,主要的两个函数是xy2lalong和laton2xy,两个函数的返回都是double型的数组。

#include <cstring>
#include <iostream>
#include "AffineModel.h"
#include "RPCGeoModel.h"

extern "C"{
    double * xy2lalong(double, double, char*);
    double * latlon2xy(double, double, char*);
}

double * xy2lalong(double x, double y,char* rpcpath)
{
    string m_prpc=rpcpath;
    //printf("%s\n",m_prpc);
    CRPCGeoModel m_prpc1;
    m_prpc1.InitRPCModel(m_prpc);
    double *latlon =new double[2];
    m_prpc1.GetLatLonByAffine(x,y, 0,latlon[0], latlon[1]); //像素坐标转经纬度
    //printf("you input %f and %f\n", latlon[0], latlon[1]);
    return latlon;
}

double * latlon2xy(double lat ,double lon,char* rpcpath)
{
    string m_prpc=rpcpath;
    //printf("%s\n",m_prpc);
    CRPCGeoModel m_prpc1;
    m_prpc1.InitRPCModel(m_prpc);
    double *xy =new double[2];
    m_prpc1.GetXY(lat, lon, 0, xy[0], xy[1]); //经纬度转像素坐标
    return xy;
}

对于多个cpp转程so文件,命令如下:

g++ -std=gnu++0x convert.cpp AffineModel.cpp CommonFunc.cpp RPCGeoModel.cpp -fPIC -shared -o libconvert.so

即将cpp文件名称都输入至终端,然后加上-fPIC -shared -o 表示生成共享库,最后加上 libXXX.so。注意一定要加上-std=gnu++0x,否则会出现如下错误。

error: ‘nullptr’ was not declared in this scope
  FILE *fp= nullptr;

之后用python调用,如下。由于c++中函数的输出是一个double的数组,那么在python的调用中也需要将函数的输出转换为相同的类型(lib.function.restype = ctypes.POINTER(ctypes.c_double)),即double的数组。

input1 = 100.0
input2 = 100.0
lib.xy2lalong.restype = ctypes.POINTER(ctypes.c_double)
a = lib.xy2lalong(ctypes.c_double(input1),ctypes.c_double(input2), strs)
print(a[0],a[1])

input1 = 9.1949132495
input2 = 118.038103295
lib.latlon2xy.restype = ctypes.POINTER(ctypes.c_double)
b = lib.latlon2xy(ctypes.c_double(input1),ctypes.c_double(input2),strs)
print(b[0],b[1])
print '***finish***' 

 

  • 4
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++调用 Python 可以使用 Python 的 C API 或者使用第三方库(如 Boost.Python)。这介绍使用 Python 的 C API 的方法。 首先需要安装 Python 和对应的开发包。以 Ubuntu 为例,可以使用以下命令安装: ``` sudo apt-get update sudo apt-get install python3-dev ``` 接下来,可以使用以下代码在 C++调用 Python: ```c++ #include <Python.h> int main() { Py_Initialize(); // 导入 Python 模块 PyObject* module = PyImport_ImportModule("example"); // 获取 Python 函数 PyObject* function = PyObject_GetAttrString(module, "example_function"); // 调用 Python 函数 PyObject* result = PyObject_CallObject(function, NULL); // 处理函数返回值 if (result != NULL) { // 如果返回值是字符串 if (PyUnicode_Check(result)) { printf("Result: %s\n", PyUnicode_AsUTF8(result)); } // 如果返回值是整数 else if (PyLong_Check(result)) { printf("Result: %ld\n", PyLong_AsLong(result)); } // 如果返回值是浮点数 else if (PyFloat_Check(result)) { printf("Result: %f\n", PyFloat_AsDouble(result)); } // 如果返回值是列表 else if (PyList_Check(result)) { Py_ssize_t size = PyList_Size(result); printf("Result: ["); for (Py_ssize_t i = 0; i < size; i++) { PyObject* item = PyList_GetItem(result, i); if (PyUnicode_Check(item)) { printf("%s", PyUnicode_AsUTF8(item)); } else if (PyLong_Check(item)) { printf("%ld", PyLong_AsLong(item)); } else if (PyFloat_Check(item)) { printf("%f", PyFloat_AsDouble(item)); } if (i < size - 1) { printf(", "); } } printf("]\n"); } Py_DECREF(result); } else { PyErr_Print(); } // 释放资源 Py_DECREF(function); Py_DECREF(module); Py_Finalize(); return 0; } ``` 以上代码假设 Python 文件名为 `example.py`,文件内容如下: ```python def example_function(): return "Hello, world!" ``` 此时在 C++调用 Python 函数 `example_function` 将会输出 `Result: Hello, world!`。 需要注意的是,使用 Python 的 C API 需要手动管理内存,需要使用 `Py_INCREF` 和 `Py_DECREF` 来增加和减少对象的引用计数,避免内存泄漏和野指针。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值