python ctypes返回零作为数组中的第一个元素?

时间:2018-07-12 14:26:05

标签: python c python-3.x numpy ctypes

我有一个使用ctypes的简单python脚本,该脚本将数组指针传递给外部函数。然后,外部函数仅意味着将相同的数组传递回python(这意味着作为简单的单元测试,以确保python和外部函数都使用相同的值)。函数原型如下:

void Array_tmp(int32_t Array[], int32_t *len, int32_t Array2[])

和python脚本看起来像这样:

from ctypes import *
import numpy as np
import numpy.ctypeslib
import os

dll = 'array_test.dll'

loadlib = cdll.LoadLibrary(dll)

arr = np.array([1,2,3,4,5])
length = len(arr)

c_arr = np.ctypeslib.as_ctypes(arr)
clen = c_int32(length)
p_clen = pointer(clen)
c_arr_ptr = cast(c_arr, POINTER(c_double))
loadlib.Array_tmp.argtypes = [type(c_arr_ptr), type(p_clen)]
loadlib.Array_tmp.restype = type(c_arr)

g = loadlib.Array_tmp(c_arr_ptr, p_clen)
print(g)
np_arr_g = np.ctypeslib.as_array(g)
print(np_arr_g)

脚本的输出如下:

<numpy.ctypeslib.c_long_Array_5 object at 0x044A7AD0>
[0 2 3 4 5]

为什么数组的第一个元素显示为零而不是一个,而其他所有元素都是正确的?

编辑:将行c_arr_ptr = cast(c_arr, POINTER(c_double))更改为c_arr_ptr = cast(c_arr, POINTER(c_int32))对输出没有影响,除了可能会稍微改变内存地址之外。

1 个答案:

答案 0 :(得分:0)

所提供的代码有很多问题。这是一个工作示例:

test.c

#include <memory.h>
#include <inttypes.h>

#define API __declspec(dllexport) /* Windows-specific export */

/* FYI: No reason for 'len' to be a pointer */
API void Array_tmp(int32_t Array[], int32_t *len, int32_t Array2[])
{
    memcpy(Array2,Array,*len * sizeof(int32_t));
}

test.py

from ctypes import *
import numpy as np

dll = CDLL('test')

# Good idea to declare the functions arguments and return type fully for error checking.
func = dll.Array_tmp
func.argtypes = POINTER(c_int32),POINTER(c_int32),POINTER(c_int32)
func.restype = None

# Create input array and output array of same size.
arr = np.array([1,2,3,4,5])
out_arr = np.zeros(arr.shape,dtype=np.int32)

# Get everything as a ctypes object
c_arr = np.ctypeslib.as_ctypes(arr)
c_out_arr = np.ctypeslib.as_ctypes(out_arr)
clen = c_int32(len(arr))

func(c_arr,byref(clen),c_out_arr)

# Original output array of zeros is now modified
print(out_arr)

输出:

[1 2 3 4 5]