问题描述
我有一个 C 函数,它使用 char 数组参数调用 go-Function。 go-Function 必须修改参数的内容。如何做到这一点?
void cFunction() {
char buffer[9] = "aaabbbccc"; // 9 is in this case correct,it is not a null-terminated-string
goFunction(buffer);
// buffer shall be modified
}
func goFunction(cBuffer *C.char) {
// how to modify 3-5?
//strncpy(cBuffer+3,"XXX")
}
编辑:更准确地说。我必须实现一个回调函数,它接受一个我必须操纵的输出参数。
void callback(char outbuffer[9]) {
goFunction(outbuffer);
}
据我了解弗兰克斯的回答,我应该做类似的事情
allocate new go buffer
convert C buffer to go buffer
manipulate go buffer
allocate new C buffer
convert go buffer to C buffer
memcpy C buffer into outbuffer
这对我的口味来说太多了分配和转换
解决方法
请参阅 Turning C arrays into Go slices 的文档以获取包含 C 数据的可索引 go 切片。
因为你是在原地修改C缓冲区数据,使用Go slice作为代理,你可以简单地将相同的缓冲区传递给回调。请注意,使用 append
可能会为您的切片分配一个新的 Go 数组,因此您需要避免它并确保事先在缓冲区中有足够的可用空间。
func goFunction(cBuffer *C.char,length int) {
slice := (*[1 << 28]C.char)(unsafe.Pointer(cBuffer))[:length:length]
// slice can now be modified using Go syntax,without pointer arithmetic
C.callback(cBuffer)
}
,
不建议在 Go 中修改 C struct,在 C 中不建议修改 Go struct,在接口 Ref1 处进行转换。
一些特殊函数通过复制数据在 Go 和 C 类型之间进行转换。在伪 Go 定义中
更多,给你一个零拷贝转换字符串的方法,Ref2。
func char2Slice(data unsafe.Pointer,len C.int) []byte {
var value []byte
sH := (*reflect.SliceHeader)(unsafe.Pointer(&value))
sH.Cap,sH.Len,sH.Data = int(len),int(len),uintptr(data)
return value
}