在Go中解析json时保留int64值

sic*_*icr 10 parsing json go

我在Go中处理一个json POST,它包含一个包含64位整数的对象数组.当使用json.Unmarshal时,这些值似乎被转换为float64,这不是很有用.

body := []byte(`{"tags":[{"id":4418489049307132905},{"id":4418489049307132906}]}`)

var dat map[string]interface{}
if err := json.Unmarshal(body, &dat); err != nil {
    panic(err)
}

tags := dat["tags"].([]interface{})

for i, tag := range tags {

    fmt.Println("tag: ", i, " id: ", tag.(map[string]interface{})["id"].(int64))

}
Run Code Online (Sandbox Code Playgroud)

有没有办法在json.Unmarshal的输出中保留原始的int64?

去上面代码的游乐场

Den*_*ret 20

解决方案1

您可以使用解码器UseNumber来解码您的数字而不会丢失:

Number类型定义如下:

// A Number represents a JSON number literal.
type Number string
Run Code Online (Sandbox Code Playgroud)

这意味着你可以轻松转换它:

package main

import (
    "encoding/json"
    "fmt"
    "bytes"
    "strconv"
)

func main() {
    body := []byte("{\"tags\":[{\"id\":4418489049307132905},{\"id\":4418489049307132906}]}")
    dat := make(map[string]interface{})
    d := json.NewDecoder(bytes.NewBuffer(body))
    d.UseNumber()
    if err := d.Decode(&dat); err != nil {
        panic(err)
    }
    tags := dat["tags"].([]interface{})
    n := tags[0].(map[string]interface{})["id"].(json.Number)
    i64, _ := strconv.ParseUint(string(n), 10, 64)
    fmt.Println(i64) // prints 4418489049307132905
}
Run Code Online (Sandbox Code Playgroud)

解决方案2

您还可以根据需要解码为特定结构:

package main

import (
    "encoding/json"
    "fmt"
)

type A struct {
    Tags []map[string]uint64 // "tags"
}

func main() {
    body := []byte("{\"tags\":[{\"id\":4418489049307132905},{\"id\":4418489049307132906}]}")
    var a A
    if err := json.Unmarshal(body, &a); err != nil {
        panic(err)
    }
    fmt.Println(a.Tags[0]["id"]) // logs 4418489049307132905
}
Run Code Online (Sandbox Code Playgroud)

就个人而言,我通常更喜欢这种感觉更结构化,更易于维护的解决方案.

警告

如果您使用JSON,因为您的应用程序部分使用JavaScript,请注意:JavaScript没有64位整数,只有一种数字类型,即IEEE754双精度浮点数.因此,您无法使用标准解析函数在JavaScript中解析此JSON而不会丢失.