在Python ctypes中使用array.array

时间:2012-05-17 00:04:16

标签: python ctypes

我必须将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_?

由于

2 个答案:

答案 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))