20

64 ビット整数を含むオブジェクトの配列を含む json POST を Go で処理しています。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))

}

json.Unmarshal の出力で元の int64 を保持する方法はありますか?

上記コードの Go Playground

4

3 に答える 3

35

解決策 1

DecoderUseNumberを使用して、数値を失うことなくデコードできます。

タイプは次のNumberように定義されます。

// A Number represents a JSON number literal.
type Number string

つまり、簡単に変換できます。

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
}

解決策 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
}

個人的には、一般的に、より構造化されていて維持しやすいこのソリューションを好みます。

注意

アプリケーションの一部が JavaScript であるために JSON を使用する場合の注意: JavaScript には 64 ビットの整数はなく、IEEE754 の倍精度浮動小数点数である 1 つの数値型しかありません。そのため、標準の解析関数を使用して損失なしに JavaScript でこの JSON を解析することはできません。

于 2013-06-05T17:43:50.950 に答える
0

これは非常に古いことに気づきましたが、これは私が最終的に使用したソリューションです

/* 
   skipping previous code, this is just converting the float 
   to an int, if the value is the same with or without what's 
   after the decimal points
*/

f := tag.(map[string]interface{})["id"].(float64)
if math.Floor(f) == f {
  fmt.Println("int tag: ", i, " id: ", int64(f))    
} else {
  fmt.Println("tag: ", i, " id: ", f)
}
于 2021-01-19T20:07:08.273 に答える