登录
首页 >  Golang >  Go问答

将 CGO 中的字符串数组转换为 GO

来源:stackoverflow

时间:2024-04-30 16:54:37 462浏览 收藏

偷偷努力,悄无声息地变强,然后惊艳所有人!哈哈,小伙伴们又来学习啦~今天我将给大家介绍《将 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))
}

经过几个小时的浏览,我找到了这个简洁的解决方案(source):

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》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>