首页 > 解决方案 > 如何通过 CGO 从 C 函数中的内部 void * 返回 []byte?

问题描述

我用以下定义包装了一个 C 函数:

int parser_shift(parser* parser, void* buffer, int length);

它从未解析字节的内部缓冲区中删除最多 length 个字节,并将它们存储在给定的缓冲区中。

现在我希望将它包装成一个 Go 函数,定义如下:

func (p *Parser) Shift() []byte {
    var buffer []byte
    // TODO:

    return buffer
}

用 CGO 完成上面的 TODO 的正确写作方式是什么?

我尝试了以下方法,但是它因错误而崩溃:“/path/to/my/program'中的错误:free():下一个大小无效(快速):0x00007f8fe0000aa0:

var buffer []byte
bufStr := C.CString(string(buffer))
defer C.free(unsafe.Pointer(bufStr))
C.parser_shift(p.Cparser, unsafe.Pointer(bufStr), C.int(8192))
buffer = []byte(C.GoString(bufStr))
return buffer

标签: gocgo

解决方案


假设您的parser_shift函数返回实际存储在缓冲区中的字节数,您可以执行以下操作:

func (p *Parser) Shift() []byte {
    var buffer [8192]byte
    parsed := int(C.parser_shift(p.Cparser, unsafe.Pointer(&buffer[0]), C.int(len(buffer))))
    return buffer[:parsed]
}

无需转换为字符串或从字符串转换,只需将您希望写入的内存传递给它即可。


推荐阅读