背景:使用 cgo 从 Golang 调用 C 函数。
我想用它有这个签名C函数:int f(int *count, char ***strs)。它将修改countand的数据strs,这就是它使用指向它们的指针的原因。的值count是 的长度strs;strs是一个字符串数组;返回值只是一个(布尔值)指示符,用于说明是否存在错误。
在golang中,我可以count通过使用成功通过和修改C.f((*C.int)(&count));通过[]string使用[]*C.char. 示例代码是这样的:
/*
#include <stdio.h>
int f(int *c, char **str) {
int i;
printf("%d\n", *c);
for (i = 0; i < *c; i++) {
printf("%s\n", str[i]);
}
*c = (*c) + 1;
return 1;
}
*/
import "C"
func go_f(strs []string) int {
count := len(strs)
c_count := C.int(count)
c_strs := make([]*C.char, count)
for index, value := range strs {
c_strs[index] = C.CString(value)
defer C.free(unsafe.Pointer(c_strs[index]))
}
err := C.f(&c_argc, (**C.char)(&c_argv[0]))
return int(err)
}
Run Code Online (Sandbox Code Playgroud)
如您所见,C 函数当前是int f(int *c, char **str),但我想要的是int f(int *c, char ***str).
这就是说:我真正想要的是在 C 中启用对字符串数组的修改(例如调整大小)并将其转回 Go 字符串切片,以便我仍然可以在 Go 中使用它。
这该怎么做?我已经搜索和试验了一段时间,但没有运气。
Go 切片在 Go 中分配,并且数据结构与 C 数组不同,因此您不能将其传递给 C 函数(cgo 也会阻止您这样做,因为切片包含 Go 指针)
您需要在 C 中分配数组,以便在 C 中操作数组。就像 C.CString 一样,您还需要跟踪释放外部数组的位置,尤其是在 C 函数可能分配新数组的情况下。
cArray := C.malloc(C.size_t(c_count) * C.size_t(unsafe.Sizeof(uintptr(0))))
// convert the C array to a Go Array so we can index it
a := (*[1<<30 - 1]*C.char)(cArray)
for index, value := range strs {
a[index] = C.CString(value)
}
err := C.f(&c_count, (***C.char)(unsafe.Pointer(&cArray)))
Run Code Online (Sandbox Code Playgroud)