4

我正在尝试在我的 Go 代码中实例化一个 C 结构。结构是这样定义的(在我无法修改的外部库中):

typedef struct {
    char field1[256];
} S1

在进行中,我这样做了:

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) 但它显然也不起作用。

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

4

2 回答 2

3

最简单的解决方案是将结构的字段定义更改为字符指针,这对于 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]代码未经测试,无法在此工作站上编译。

于 2014-08-15T13:44:10.000 回答
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)),
}
于 2014-08-15T13:51:23.147 回答