当前位置: > > > > 将 CGO 中的字符串数组转换为 GO
来源:stackoverflow
2024-04-30 16:54:37
0浏览
收藏
偷偷努力,悄无声息地变强,然后惊艳所有人!哈哈,小伙伴们又来学习啦~今天我将给大家介绍《将 CGO 中的字符串数组转换为 GO》,这篇文章主要会讲到等等知识点,不知道大家对其都有多少了解,下面我们就一起来看一吧!当然,非常希望大家能多多评论,给出合理的建议,我们一起学习,一起进步!
问题内容
我可以转换从 go 中的 c (cgo) 函数返回的字符串数组 (char**) 吗? 下面的代码可以编译并运行,但我无法遍历字符串列表。
我什至不确定它是否违反了“传递指针”的规则: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)) }
经过几个小时的浏览,我找到了这个简洁的解决方案():
list := c.getlist() length := 2 slice := make([]string, length) for _, v := range unsafe.slice(list, length) { slice = append(slice, c.gostring(v)) }
对于长度猜测问题,您可以让 c 函数获取指向字符串数组的指针并返回大小。然后它看起来像这样:
var list **C.char length := C.getList(&list) slice := make([]string, length) for _, v := range unsafe.Slice(list, length) { slice = append(slice, C.GoString(v)) }
今天关于《将 CGO 中的字符串数组转换为 GO》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注公众号!