cgo - 如何将字符串转换为C固定字符数组

4

我希望在我的Go代码中实例化一个C结构体。这个结构体是这样定义的(在一个我不能修改的外部库中):

typedef struct {
    char field1[256];
} S1

在Go中,我这样做:
func myfunc(mystr string){
    // We need to convert mystr from string to char array
    cStr := C.CString(mystr)
    defer C.free(unsafe.Pointer(cStr)

    // Should work now
    s1 := &C.S1{field1: cStr}

    // Do something with s1...
}

但是它无法编译,因为:

在字段值中不能使用cStr (类型*C.char)作为类型[256]C.char。

我尝试强制使用([256]C.char)(cStr),但显然也不起作用。

有没有办法实现我想做的事情?


使用类似 copy(sq.fields[:], mystr) 的方法。 - fuz
2个回答

3
最简单的解决方案是将您的结构体字段定义更改为 char 指针,这在 C 中的字符串中非常标准:
typedef struct {
    char *field1;
} S1

更加复杂的解决方案是[1]:
arr := [256]C.char{}

for i := 0; i < len(mystr) && i < 255; i++ { // leave element 256 at zero
    arr[i] = C.char(mystr[i])
}

s1 := &C.S1{field1: arr}

[1] 代码未经测试,无法在此工作站上编译。


很遗憾,我无法修改这个结构体。它来自外部库。 - David23
1
您提供的代码有效:)。非常感谢! - David23
你可以使用内置的 copy() 函数来实现这个功能。 - fuz

3

很不幸,Go语言中没有方便的处理[size]C.char作为字符串的方法(我记得曾经有一个提案来添加这个功能...)

在我的代码中,我选择手动将字符串写入结构体中,而不是直接处理它,可以使用以下类似的方式:

func strCopy(dest *[maxTextExtent]C.char, src []byte) {
    for i, c := range src {
        dest[i] = C.char(c)
    }
    // This is C, we need to terminate the string!
    dest[len(src)] = 0
}

我曾经用过的处理方式,这种方式不够安全,是

s1 := &C.S1{
    field1: *(*[256]C.char)(unsafe.Pointer(cStr)),
}

@OneOfOne:你可能会覆盖一个已经在C中分配的数组。 - JimB
@tomwilde:我知道,但在Go中会正确地引发panic(在C中你绝对想要检查) - JimB
谢谢您的回答,它很有效 :)。我想接受两个答案,但我认为我会接受@tomwilde的,因为他的声望较低,而且最终我使用了他的代码。 再次感谢! - David23
1
@David23:没问题 :),不过请检查最新的编辑。我并不是在鼓励你使用它,但你可能会对直接分配它的工作方式感兴趣。 - JimB
@JimB:谢谢你 :) 我尝试过类似的事情(没有成功),但最后停下来了,因为使用越来越“不安全”,就会变得越来越不安全 ;)。 - David23
显示剩余3条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接