13

我正在尝试使用 ctypes 在 python 中创建一个 char * 数组以传递给库以填充字符串。我期待 4 个字符串返回,每个字符串的长度不超过 7 个字符。

我的 py 代码看起来像这样

测试库.py

from ctypes import *
primesmile = CDLL("/primesmile/lib.so")

getAllNodeNames = primesmile.getAllNodeNames
getAllNodeNames.argtypes = [POINTER(c_char_p)]
results = (c_char_p * 4)(addressof(create_string_buffer(7)))
err = getAllNodeNames(results)

库文件

void getAllNodeNames(char **array){
    DSL_idArray nodes; //this object returns const char * when iterated over
    network.GetAllNodeIds(nodes);
    for(int i = 0; i < (nodes.NumItems()); i++){
    strcpy(array[i],nodes[i]);
    }
}

当我尝试运行此代码时,我不断收到分段错误。我已经从 C 创建了一个完美运行的测试,但在 Python 中我必须错误地设置指针数组或其他东西。它似乎到达了循环中的第二个节点,然后出现了问题,正如我从将数据吐出到命令行中看到的那样。任何见解将不胜感激。

4

3 回答 3

12

以下代码有效:

测试.py:

import ctypes
lib = ctypes.CDLL("./libtest.so")
string_buffers = [ctypes.create_string_buffer(8) for i in range(4)]
pointers = (ctypes.c_char_p*4)(*map(ctypes.addressof, string_buffers))
lib.test(pointers)
results = [s.value for s in string_buffers]
print results

test.c(编译为 libtest.so gcc test.c -o libtest.so -shared -fPIC):

#include <string.h>
void test(char **strings) {
    strcpy(strings[0],"this");
    strcpy(strings[1],"is");
    strcpy(strings[2],"a");
    strcpy(strings[3],"test!");
}

正如 Aya 所说,您应该确保有终止零的空间。但我认为您的主要问题是字符串缓冲区被垃圾收集或类似的东西,因为不再直接引用它。或者当没有为字符串缓冲区存储引用时,其他东西会在字符串缓冲区的创建过程中造成麻烦。例如,这会导致四次相同的地址而不是不同的地址:

import ctypes
pointers = [ctypes.addressof(ctypes.create_string_buffer(8)) for i in range(4)]
print pointers
于 2013-05-22T19:36:57.613 回答
4

在这一行:

results = (c_char_p * 4)(addressof(create_string_buffer(7)))

您正在创建一个 7 字节的缓冲区,然后尝试使用它来保存 4 个字符指针(每个可能是 4 个字节),然后还将 4 个 8 字节字符串复制到它可能碰巧指向的随机地址中。您需要为每个字符串分配一个缓冲区,并分配指针数组。像这样的东西:

s = []
for i in range(4):
    s[i] = create_string_buffer(8)
results = (c_char_p * 4)(s);
于 2013-05-22T19:33:55.487 回答
2

我不能(轻松)测试代码,但根据你所说的,我的猜测是问题出在这条线上......

results = (c_char_p * 4)(addressof(create_string_buffer(7)))

根据 Python 文档create_string_buffer()...

init_or_size必须是指定数组大小的整数,或者是用于初始化数组项的字符串。

如果将字符串指定为第一个参数,则缓冲区将比字符串的长度大一个项目,以便数组中的最后一个元素是 NUL 终止字符。如果不应使用字符串的长度,则可以将整数作为第二个参数传递,该参数允许指定数组的大小。

...这意味着它正在创建 a char[7],但 astrcpy()将尝试复制NULL字符串的终止符,因此如果您的最大“节点名称”长度为七个字符,则您需要 achar[8]来保存NULL,尽管您可能会使用memcpy(array[i], nodes[i], 7)而不是strcpy().

无论哪种方式,使用它可能是最安全的create_string_buffer(8)

于 2013-05-22T19:25:30.920 回答