在 GO 中从 CGO 转换字符串数组

convert array of strings from CGO in GO

我可以转换从 Go 中的 C (cgo) 函数返回的字符串数组 (char**) 吗? 下面的代码编译并运行,但我无法遍历字符串列表。
我什至不确定它是否违反了 "passing pointers" 的规则:https://golang.org/cmd/cgo/ 任何想法都会有所帮助,自从我用 C 编写代码以来已经有好几年了!提前致谢!

package main
/*
#include "stdlib.h"

char** getlist ()
{
    char **array = NULL;
    array = (char**)realloc(array, 2*sizeof(*array));
    array[0]="HELLO";
    array[1]="WORLD";
    return array;
}

*/
import "C"

import (
    "log"
    "unsafe"
)

func main() {
    list := C.getlist();
    log.Printf("\n========\n C.getList()=%s", list)
    ulist := unsafe.Pointer(list)
    log.Printf("\n========\nulist=%s", ulist)
}

为了在 Go 中遍历字符串,您需要将数组转换为 Go 切片。我们可以在这里跳过分配,直接转换它(你的例子静态地将长度设置为 2,但实际上你可能有这个大小的另一个来源)

cSlice := (*[1 << 28]*C.char)(unsafe.Pointer(list))[:2:2]

我们可以直接迭代这个,然后使用C.GoString函数来转换C字符串。这样存储起来更安全,因为它会将数据复制到 Go 内存中,但是如果这个切片非常大,我们可以使用与上面相同的不安全转换来保存分配,尽管您首先需要找到每个字符串的长度。

var slice []string
for _, s := range (*[1 << 28]*C.char)(unsafe.Pointer(list))[:2:2] {
    slice = append(slice, C.GoString(s))
}