登录
首页 >  Golang >  Go问答

为何在将 Go 字节传递给 C 时,这段代码中的字节长度与 cgo 中的字节长度不一致?

来源:stackoverflow

时间:2024-02-25 11:45:24 213浏览 收藏

本篇文章向大家介绍《为何在将 Go 字节传递给 C 时,这段代码中的字节长度与 cgo 中的字节长度不一致?》,主要包括,具有一定的参考价值,需要的朋友可以参考一下。

问题内容

将 golang 字节传递给 c 时,字节长度不匹配。

生成的 strlen(key) 和 keylen 不匹配。

使用“go build file.go”构建

您可以在此处下载下面的 go 文件: https://pastebin.com/raw/hnmfxjkq <-有谁知道为什么我的 cgo 调用 []bytes to c 有错误?为什么 strlen 不匹配?

预期输出应该具有相同的密钥长度。有时有效,有时无效。

package main
/*
#include 
#include 
#include 
#include 
#include 
#include 

void replaceK(void *key, size_t keylen, void *value, size_t valuelen);
void replaceK(void *key, size_t keylen, void *value, size_t valuelen)
{
        printf("replaceK : key = %s = %lu = %lu = %s = %lu\n",(char*) key,keylen,strlen(key),(char*) value,valuelen);
        if (keylen != strlen(key)){
                printf("ERROR!!! keylen : %lu != strlen(key) : %lu!!!\n",keylen,strlen(key));
                exit(1);
        }
}
*/
import "C"
import (
        "fmt"
        "unsafe"
        "math/rand"
        "time"
)
func Set(key,value []byte) {
        cKey := cByteSlice(key)
        cValue := cByteSlice(value)
        C.replaceK(unsafe.Pointer(cKey),C.ulong(len(key)),unsafe.Pointer(cValue),C.ulong(len(value)))
        C.free(unsafe.Pointer(cKey))
        C.free(unsafe.Pointer(cValue))
}
func byteToChar(b []byte) *C.char {
        var c *C.char
        if len(b) > 0 {
                c = (*C.char)(unsafe.Pointer(&b[0]))
        }
        return c
}
var letterRunes = []byte("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ")
func RandStringRunes(n int) []byte {
        randNum := rand.Intn(n)+1
        b := make([]byte, randNum)
        for i := range b {
                b[i] = letterRunes[rand.Intn(len(letterRunes))]
        }
        return b
}
func cByteSlice(b []byte) *C.char {
        var c *C.char
        if len(b) > 0 {
                c = (*C.char)(C.CBytes(b))
        }
        return c
}
func main() {
        rand.Seed(time.Now().UnixNano())
        var key []byte
        var value []byte
        for i:=0;i<10000000;i++ {
                key = RandStringRunes(10)
                value = RandStringRunes(20)
                randnum := 1
                if randnum == 1 {
                        fmt.Printf(">>> = %s = %s\n",key, value)
                        Set(key,value)
                }
        }
}

正确答案


C strlen 函数用于以 null 结尾的字符串,而不是指针+长度字符串。

您可以使用 C printf 函数使用 %.*s 而不是 %s 来打印指针+长度字符串。由于 Go string[]byte 变量都使用指针+长度编码,这可能是打印从 Go 获取的字符串的正确方法。

今天关于《为何在将 Go 字节传递给 C 时,这段代码中的字节长度与 cgo 中的字节长度不一致?》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

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