我试图用ctype从C++调用python函数(回调)。python函数接受一个ints数组作为参数。我需要在python中访问这个数组。
C码
typedef void (*FooFunc)(int*, int*);
void runfoo(FooFunc foo_func) {
int dims[5] = {3, 5, 1, 1, 1};
foo_func(dims, dims);
}python代码
def foo(dims1, dims2):
x = dims1[0]
y = dims1[1]
z = dims1[2]
print(x, y, z)
x = dims2[0]
y = dims2[1]
z = dims2[2]
print(x, y, z)
libc = ctypes.CDLL('./libpath.so')
protofoo = ctypes.CFUNCTYPE(None, ctypes.c_int * 5, ctypes.POINTER(ctypes.c_int))
libc.runfoo(protofoo(foo))它给出(-333106720,32766,-333106720) (3,5,1)。
ctypes.c_int *5和ctypes.POINTER(ctypes.c_int)在原型中有什么区别?
发布于 2019-07-14 20:41:02
ctypes.POINTER(ctypes.int)等同于int*。
ctypes.c_int * 5是一种与int arr[5]类型等价的类型。它的实现不会作为像C这样的参数衰减到指针,所以当使用它作为参数时,它希望在堆栈上通过值传递一个数组。下面通过在C中以十六进制打印dims的地址进行了测试,您可以看到它的64位指针值是显示为十六进制时返回到python的前两个32位元素:
test.c
#include <stdio.h>
typedef void (*FooFunc)(int*, int*);
__declspec(dllexport) void runfoo(FooFunc foo_func) {
int dims[5] = {3, 5, 1, 1, 1};
printf("%p\n",dims);
foo_func(dims, dims);
}test.py
import ctypes
def foo(dims1, dims2):
x = dims1[0]
y = dims1[1]
z = dims1[2]
print(f'{x:08X} {y:08X} {z:08X}')
x = dims2[0]
y = dims2[1]
z = dims2[2]
print(f'{x:08X} {y:08X} {z:08X}')
libc = ctypes.CDLL('test')
protofoo = ctypes.CFUNCTYPE(None, ctypes.c_int * 5, ctypes.POINTER(ctypes.c_int))
libc.runfoo(protofoo(foo))输出
000000FD6E1EED00
6E1EED00 000000FD 6E1EED00
00000003 00000005 00000001https://stackoverflow.com/questions/57027124
复制相似问题