如果要使用第一种方法,则需要在函数调用参数之外创建切片,并避免临时分配的切片标头或参数中的外部结构,因此cgo检查不会将其视为存储在其中的指针走.
b := buf.Bytes()
rc := C.the_function(unsafe.Pointer(&b[0]), C.int(buf.Len()))
C.CString方法将更安全,因为数据被复制到C缓冲区,因此没有指向Go内存的指针,并且bytes.Buffer后面的切片不可能被修改或超出范围.您将要转换整个字符串,而不仅仅是第一个字节.这种方法确实需要分配和复制两次,但是如果数据量很小,那么与cgo调用本身的开销相比,这可能不是一个问题.
str := buf.String()
p := unsafe.Pointer(C.CString(str))
defer C.free(p)
rc = C.the_function(p, C.int(len(str)))
如果在该解决方案中不接受2个数据副本,则有第三个选项,您自己对C缓冲区进行malloc,并将一个副本复制到该缓冲区中:
p := C.malloc(C.size_t(len(b)))
defer C.free(p)
// copy the data into the buffer, by converting it to a Go array
cBuf := (*[1 << 30]byte)(p)
copy(cBuf[:], b)
rc = C.the_function(p, C.int(buf.Len()))
但是对于后两种选择,不要忘记释放malloc指针.