python ctypes中的多维char数组(字符串数组)


问题内容

我正在尝试使用ctypes将字符数组的数组传递给C函数。

void cfunction(char ** strings)
{
 strings[1] = "bad"; //works not what I need.
 strings[1][2] = 'd'; //this will segfault.
 return;
}

char *input[] = {"foo","bar"};
cfunction(input);

由于我抛出的数组始终是静态定义的,因此我只更改了函数声明和输入参数,如下所示:

void cfunction(char strings[2][4])
{
 //strings[1] = "bad"; //not what I need.
 strings[1][2] = 'd'; //what I need and now it works.
 return;
}

char input[2][4] = {"foo","bar"};
cfunction(input);

现在,我遇到了如何在python中定义此多维字符数组的问题。我以为会这样:

import os
from ctypes import *
libhello = cdll.LoadLibrary(os.getcwd() + '/libhello.so')
input = (c_char_p * 2)()
input[0] = create_string_buffer("foo")
input[1] = create_string_buffer("bar")
libhello.cfunction(input)

这给了我TypeError: incompatible types, c_char_Array_4 instance instead of c_char_p instance。如果我将其更改为:

for i in input:
 i = create_string_buffer("foo")

然后我得到分割错误。同样,这似乎是构建2d数组的错误方法,因为如果我打印输入,我会看到None

print input[0]
print input[1]

# outputs None, None instead of "foo" and "foo"

我还遇到了在C文件中使用#DEFINE MY_ARRAY_X 2#DEFINE MY_ARRAY_Y 4保持数组维数直接的问题,但是我不知道从libhello.js中获取这些常量的好方法,因此python在构造数据类型时可以引用它们。


问题答案:

使用类似

input = ((c_char * 4) * 2)()
input[0].value = "str"
input[0][0] == "s"
input[0][1] == "t" # and so on...

简单用法:

>>> a =((c_char * 4) * 2)()
>>> a
<__main__.c_char_Array_4_Array_2 object at 0x9348d1c>
>>> a[0]
<__main__.c_char_Array_4 object at 0x9348c8c>
>>> a[0].raw
'\x00\x00\x00\x00'
>>> a[0].value
''
>>> a[0].value = "str"
>>> a[0]
<__main__.c_char_Array_4 object at 0x9348c8c>
>>> a[0].value
'str'
>>> a[0].raw
'str\x00'
>>> a[1].value
''
>>> a[0][0]
's'
>>> a[0][0] = 'x'
>>> a[0].value
'xtr'