我必须将Python程序与C库连接。我需要调用的特定函数接受一个数组并返回一个double。以下函数具有相同的签名,比我自己的更容易理解:
double sum(double * array, const int length) {
double total = 0;
int i;
for (i=0; i<length; i++) {
total += array[i];
}
return total;
}
我目前的解决方案是:
import ctypes
lib = ctypes.CDLL(library_name)
l = 10
arr = tuple(range(l))
lib.sum.restype = ctypes.c_double
values = (ctypes.c_double * l)(*arr)
ret = lib.sum(values, l)
但是我在代码中使用了很多数组模块,在我看来,将它们与C代码一起使用应该更直接,因为它是一个类型化的数组。所以我尝试直接用数组提供C函数,但它不起作用。为了使它工作,我像这样包装数组:
class Array(array):
@property
def _as_parameter_(self):
return (TYPES[self.typecode] * len(self))(*self)
其中TYPES将数组中的类型代码映射到ctypes类型:
TYPES = {'c': ctypes.c_char,
'b': ctypes.c_byte,
'B': ctypes.c_ubyte,
'?': ctypes.c_bool,
'h': ctypes.c_short,
'H': ctypes.c_ushort,
'i': ctypes.c_int,
'I': ctypes.c_uint,
'l': ctypes.c_long,
'L': ctypes.c_ulong,
'q': ctypes.c_longlong,
'Q': ctypes.c_ulonglong,
'f': ctypes.c_float,
'd': ctypes.c_double}
有没有办法用不会创建另一个数组的东西替换_as_parameter_?
由于
答案 0 :(得分:4)
使用array.buffer_info()获取地址和长度,并将地址cast()传递给POINTER(c_double):
from array import array
buf = array("d", range(101))
addr, count = buf.buffer_info()
print lib.sum(cast(addr, POINTER(c_double)), count)
答案 1 :(得分:0)
并不是说这是一个更好的答案,但是如果你使用numpy则完全正确,它只是有点不同,因为numpy可以为你处理演员表。
import numpy as np
data = np.arange(101, dtype=ctypes.c_double) # making this match seems important sometimes
print lib.sum(data.ctypes.data_as(ctypes.POINTER(ctypes.c_double)), len(data))