如何将C数组的指针转换为python数组

发布于 2021-01-29 18:36:04

我有一个使用ctypes调用Python的C++回调函数。该函数的参数是一个指向double和元素数量数组的指针。

元素很多,大约2,000,000。我需要将此发送到scipy函数。

C ++原型是:

bool (*ptsetDataSource)(double*, long long);

以下是python代码:

CPF_setDataSource = CFUNCTYPE(c_bool, POINTER(c_double),c_longlong)
CPF_setSelection= CFUNCTYPE(c_bool,c_char_p, c_longlong,c_longlong)
CPF_ResetSequence = CFUNCTYPE(c_bool)

def setDataSource(Data, DataLength):
    Datalist=[0.0]*100
    for i in range(0,100):
        Datalist[i]=Data[i]

    print Datalist
    return True

问题是打印数据列表返回:

[0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0]

这是不正确的(在c ++端检查时,数据中填充了许多其他数字。

另外,如果我使用此代码将数据转换为python列表,则会在分配步骤中锁定计算机。

无论如何,是否要从C ++数组加载数据,然后将其转换为适合scipy的数组?

关注者
0
被浏览
266
1 个回答
  • 面试哥
    面试哥 2021-01-29
    为面试而生,有面试问题,就找面试哥。

    如果Data(c_double*DataLength.value)数组,则可以:

    a = np.frombuffer(Data) # no copy. Changes in `a` are reflected in `Data`
    

    如果Data为,则POINTER(c_double)可以使用获取numpy数组numpy.fromiter()。它与您的问题相同,但速度更快:

    a = np.fromiter(Data, dtype=np.float, count=DataLength.value) # copy
    

    要从POINTER(c_double)实例创建一个numpy数组而不进行复制,您可以使用.from_address()方法:

    ArrayType = ctypes.c_double*DataLength.value
    addr = ctypes.addressof(Data.contents)
    a = np.frombuffer(ArrayType.from_address(addr))
    

    要么

    array_pointer = ctypes.cast(Data, ctypes.POINTER(ArrayType))
    a = np.frombuffer(array_pointer.contents)
    

    两种方法都将POINTER(c_double)实例转换为,(c_double*DataLength)然后再将其传递给numpy.frombuffer()

    基于Cython的解决方案

    无论如何,是否要从C ++数组加载数据,然后将其转换为适合scipy的数组?

    这是Python的C扩展模块(用Cython编写),提供了C API转换功能:

    cimport numpy as np
    np.import_array() # initialize C API to call PyArray_SimpleNewFromData
    
    cdef public api tonumpyarray(double* data, long long size) with gil:
        if not (data and size >= 0): raise ValueError
        cdef np.npy_intp dims = size
        #NOTE: it doesn't take ownership of `data`. You must free `data` yourself
        return np.PyArray_SimpleNewFromData(1, &dims, np.NPY_DOUBLE, <void*>data)
    

    可以与ctypes以下方式一起使用:

    from ctypes import (PYFUNCTYPE, py_object, POINTER, c_double, c_longlong,
                        pydll, CFUNCTYPE, c_bool, cdll)
    
    import pointer2ndarray
    tonumpyarray = PYFUNCTYPE(py_object, POINTER(c_double), c_longlong)(
        ("tonumpyarray", pydll.LoadLibrary(pointer2ndarray.__file__)))
    
    @CFUNCTYPE(c_bool, POINTER(c_double), c_longlong)
    def callback(data, size):
        a = tonumpyarray(data, size)
        # call scipy functions on the `a` array here
        return True
    
    cpplib = cdll.LoadLibrary("call_callback.so") # your C++ lib goes here
    cpplib.call_callback(callback)
    

    哪里call_callback是:void call_callback(bool (*)(double *, long long))



知识点
面圈网VIP题库

面圈网VIP题库全新上线,海量真题题库资源。 90大类考试,超10万份考试真题开放下载啦

去下载看看