python调用动态链接库的基本过程详解

作者:fariver 时间:2023-05-31 13:24:00 

动态链接库在Windows中为.dll文件,在linux中为.so文件。以linux平台为例说明python调用.so文件的使用方法。
本例中默认读者已经掌握动态链接库的生成方法,如果不太清楚的可以参考动态链接库的使用

调用上例动态链接库的使用中的sum.so


import ctypes
so = ctypes.CDLL('./sum.so')

print "so.sum(50) = %d" % so.sum(50)
so.display("hello world!")
print "so.add() = %d" % so.add(ctypes.c_float(2), ctypes.c_float(2010))

output


so.sum(50) = 1275
hello world!
so.add() = 2012

注意:

  • 如果python在调用C函数内部出现了问题,系统不会提示具体出现什么问题,只会提示"segmentation fault"。所以最好是先用C语言调用该动态库验证没有问题了再提供给python调用。

  • python传参给C函数时,可能会因为python传入实参与C函数形参类型不一致会出现问题( 一般int, string不会有问题,float要注意 )。这时需要在python调用时传入的实参做一个类型转换(见so.add(float, float)函数的调用)。转换方式见下表:

python调用动态链接库的基本过程详解

数组的传入传出

如果将python中list传入C函数数组,则需要提前转换。


import ctypes
pyarray = [1, 2, 3, 4, 5]
carrary = (ctypes.c_int * len(pyarray))(*pyarray) //有点类似malloc的方式生成carray
print so.sum_array(carray, len(pyarray))

refer

如果如果需要将C array返回python,需要提前把array传入,然后在C函数中修改,返回时再把c array转换为np.array


pyarray = [1,2,3,4,5,6,7,8]
carray = (ctypes.c_int*len(pyarray))(*pyarray)
so.modify_array(carray, len(pyarray))
print np.array(carray)

output

[10 20 30 40 50 60 70 80]

也可以用形参方式提前定义函数接口,然后再传入numpy结构


import ctypes
import numpy as np
from numpy.ctypeslib import ndpointer
so = ctypes.CDLL('./sum.so')
pyarray = np.array([1,2,3,4,5,6,7,8], dtype="int32")
fun = so.modify_array
fun.argtypes = [ndpointer(ctypes.c_int), ctypes.c_int]
fun.restype = None
fun(pyarray, len(pyarray))
print np.array(pyarray)

注意:numpy中的数据类型指定很重要,即dtype的设定

图片的传入传出

转递数据域

背景知识:

python中的opencv图片是用numpy的方式保存的,而opencv3 C语言的图片数据结构为cvMat (IplImage已经逐弃用)
所以需要把python中numpy图片转换为ctypes.POINTER(ctypes.c_ubyte)的指针转入其数据域,再将其行列信息传入,就可以在C中从最底层初始化一个CvMat,如果要初始化一个别数据结构的图片也是同理(如darknet的image,caffe的blob)
python numpy image 转换为 C pointer的方法


python_frm.ctypes.data_as(C.POINTER(ctypes.c_ubyte))

注意:传入numpy image前一定要确保numpy image是numpy array数据类型

比如我遇到的bug


image = cv2.imread("xxx.jpg");

image传入ctypes_so.fun之中图片是有效的,但


image = cv2.imread("xxx.jpg");

这时候进入ctypes_so.fun的图片会变成一个乱码

即,crop之后的numpy image的type虽然也为numpy array,但实际传入的image data却不正确

解决方法:

无论是何种方式得到的numpy image,都强行转换为numpy array,再传入ctypes_so.fun


image = numpy.array(image)

可以解决这个bug

refence

如果使用opencv2 可以考虑直接将numpy image转换为IplImage

opencv3 python已经不支持cv2.cv的函数了

但Opencv2可能还可以尝试以下方法

numpy image to iplimage

python调用C++中的类

因为python不能直接调用C++中的类,所以必须把C++中的类转换为C的接口

转换原则

  • 所有的C++关键字及其特有的使用方式均不能出现在.h文件里,.h中仅有C函数的包装函数声明

  • 在class.cpp中实现对类的成员函数接口转换的函数,包括对类内成员的读写函数get() and set()

  • 如果要在包装函数中要实例化对象,尽量用new constructor()的将对象的内存实例化在堆中,否则对象会被析构

  • 记得在所有包含函数声明的文件中加入以下关键字,声明该函数为C函数,否则该函数的符号不会记录在二进制文件中


#ifdef __cplusplus
extern "C" {
#endif
xxxxxx function declaration xxxxx
#ifdef __cplusplus
}
#endif

code
refer
refer

来源:https://www.cnblogs.com/fariver/p/6573112.html

标签:python,动态链接库
0
投稿

猜你喜欢

  • 将不规则的Python多维数组拉平到一维的方法实现

    2023-11-05 19:47:15
  • 如何选择合适的MySQL存储引擎

    2009-02-04 13:02:00
  • vue 自定义指令directive的使用场景

    2024-05-13 09:09:08
  • 详解如何用SQLyog来分析MySQL数据库

    2008-10-13 12:35:00
  • Python PyQt5实现的简易计算器功能示例

    2022-01-22 12:03:19
  • IE下,事件触发那点破烂事儿

    2009-04-27 12:31:00
  • python更新列表的方法

    2021-12-21 11:38:39
  • pymysql 插入数据 转义处理方式

    2024-01-23 08:43:29
  • Numpy 中的矩阵求逆实例

    2022-11-15 16:51:12
  • Python实现GUI学生管理系统的示例代码

    2022-06-01 01:47:56
  • python 读txt文件,按‘,’分割每行数据操作

    2022-11-11 08:04:17
  • numpy 产生随机数的几种方法

    2021-08-28 02:33:54
  • python找出列表中大于某个阈值的数据段示例

    2022-05-17 17:21:05
  • vue+elementUI动态生成面包屑导航教程

    2024-05-02 17:12:26
  • python ChainMap的使用详解

    2023-10-31 18:02:40
  • 从事设计行业的十年

    2008-04-01 09:44:00
  • JavaScript程序执行顺序问题总结

    2010-01-29 13:06:00
  • 详解SQLServer和Oracle的分页查询

    2024-01-21 10:11:39
  • Go秒爬博客园100页新闻

    2024-04-26 17:15:49
  • 解决pyshp UnicodeDecodeError的问题

    2021-08-01 10:17:39
  • asp之家 网络编程 m.aspxhome.com