程序开发 · 2023年12月21日

整数溢出-golang

当前位置: > > > > 整数溢出-golang

来源:stackoverflow
2024-04-21 23:24:38
0浏览
收藏

怎么入门Golang编程?需要学习哪些知识点?这是新手们刚接触编程时常见的问题;下面就来给大家整理分享一些知识点,希望能够给初学者一些帮助。本篇文章就来介绍《整数溢出-golang》,涉及到,有需要的可以收藏一下

问题内容

我遇到一些整数溢出错误。 我有一个使用 golang 和 go-micro 作为微服务框架构建的微服务应用程序。 我使用 nats 作为消息代理。

我的微服务有效负载的格式为

map[string]interface{}

当我发布包含 uint64 的有效负载时,会出现问题,例如

var id uint64 = 512281913614499841

 message := map[string]inteface{}
 message["id"] = id

(这是由 cockroachdb 生成的唯一 id),当订阅者以字节形式接收此消息并将其解组为 uint64 时,我意识到发生了溢出,并且该值变为

512281913614499840 //this should be 512281913614499841

注意末尾的 0 而不是 1

我创建了 2 个函数(overflowerror 和 nooverflow) – 见下文。

函数overflowerror模拟导致溢出的代码

并且 nooverflow 打印正确的结果,因为我使用这种格式的有效负载map[string][struct]而不是map[string]interface

type usertype struct {
    email string `json:"email"`
    id    int64  `json:"id"`
}
func overflowerror() {

        var id int64 = 512281913614499841

        user := usertype{
            email: "example",
            id:    id,
        }

        message := map[string]interface{}{
            "data": user,
        }

        //mashal to byte to simulate service payload
        servicepayload, err := json.marshal(message)

        if err != nil {
            log.println(err)
        }

        var receivedmessage map[string]interface{}

        json.unmarshal(servicepayload, &receivedmessage)

        var myuser usertype

        mapstructure.decode(receivedmessage["data"], &myuser)

        log.println("---receivedmessage:", myuser.id) //prints 512281913614499840 - incorrect
    }

无溢出

type UserType struct {
    Email string `json:"email"`
    ID    int64  `json:"id"`
}
func noOverflow() {

        var id int64 = 512281913614499841

        message := map[string]UserType{}

        message["data"] = UserType{
            Email: "example",
            ID:    id,
        }

        byteMessage, err := json.Marshal(message)

        if err != nil {
            log.Println(err)
        }

        var msgMap map[string]UserType

        json.Unmarshal(byteMessage, &msgMap)

        log.Println("---myUser:", msgMap["data"].ID) // prints 512281913614499841 - correct
    }

为了避免大量代码重写,我留下了第一个选项,我在 overflowerror 函数中模拟了该选项,并在上面进行了解释

这个问题有解决办法吗?

解决方案

这不是整数溢出。如 ,问题实际上是 JSON 数字倾向于表示为浮点数,而 float32 的位数太少,无法保存正确的值。即使 Go 默认使用的 float64 也还不够。

查看 (不仅仅是已接受的答案)和 的所有答案。或者,查看旧问题 及其已接受的答案。

根据您需要哪种互操作性,您可以继续将它们解码为数字,但使用更复杂的解码器和/或自定义解组器,或者只是将其编码为 JSON 字符串。实际上,当跨 Go + Python + TypeScript + 各种 C++ 库使用 JSON 时,我们1诉诸于字符串编码。我们最终发现这要可靠得多。 (我们不必对 32 位整数执行此操作,只需对 64 位整数执行此操作。)

要使用非默认解码器,请特别参阅 及其 选项。

1这里的“我们”不是皇家的“我们”,而是 $job,除了 TypeScript 之外,我接触过所有这些。

好了,本文到此结束,带大家了解了《整数溢出-golang》,希望本文对你有所帮助!关注公众号,给大家分享更多Golang知识!