web-dev-qa-db-fra.com

Comment décoder JSON avec le type converti de chaîne en float64 dans Golang?

J'ai besoin de décoder une chaîne JSON avec le nombre flottant comme:

{"name":"Galaxy Nexus", "price":"3460.00"}

J'utilise le code Golang ci-dessous:

package main

import (
    "encoding/json"
    "fmt"
)

type Product struct {
    Name  string
    Price float64
}

func main() {
    s := `{"name":"Galaxy Nexus", "price":"3460.00"}`
    var pro Product
    err := json.Unmarshal([]byte(s), &pro)
    if err == nil {
        fmt.Printf("%+v\n", pro)
    } else {
        fmt.Println(err)
        fmt.Printf("%+v\n", pro)
    }
}

Lorsque je l'exécute, obtenez le résultat:

json: cannot unmarshal string into Go value of type float64
{Name:Galaxy Nexus Price:0}

Je veux savoir comment décoder la chaîne JSON avec le type convert.

66
yanunon

La réponse est considérablement moins compliquée. Ajoutez simplement à l'interpeter JSON qu'il s'agit d'une chaîne flottante codée avec ,string (notez que je n'ai changé que la définition de Price):

package main

import (
    "encoding/json"
    "fmt"
)

type Product struct {
    Name  string
    Price float64 `json:",string"`
}

func main() {
    s := `{"name":"Galaxy Nexus", "price":"3460.00"}`
    var pro Product
    err := json.Unmarshal([]byte(s), &pro)
    if err == nil {
        fmt.Printf("%+v\n", pro)
    } else {
        fmt.Println(err)
        fmt.Printf("%+v\n", pro)
    }
}
136
Dustin

Juste pour vous faire savoir que vous pouvez le faire sans Unmarshal et utiliser json.decode. Voici Go Playground

package main

import (
    "encoding/json"
    "fmt"
    "strings"
)

type Product struct {
    Name  string `json:"name"`
    Price float64 `json:"price,string"`
}

func main() {
    s := `{"name":"Galaxy Nexus","price":"3460.00"}`
    var pro Product
    err := json.NewDecoder(strings.NewReader(s)).Decode(&pro)
    if err != nil {
        fmt.Println(err)
        return
    }
    fmt.Println(pro)
}
8
Salvador Dali

Passer une valeur entre guillemets donne l'impression que c'est une chaîne. Changement "price":"3460.00" à "price":3460.00 et tout fonctionne bien.

Si vous ne pouvez pas supprimer les guillemets, vous devez l'analyser par vous-même, en utilisant strconv.ParseFloat:

package main

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

type Product struct {
    Name       string
    Price      string
    PriceFloat float64
}

func main() {
    s := `{"name":"Galaxy Nexus", "price":"3460.00"}`
    var pro Product
    err := json.Unmarshal([]byte(s), &pro)
    if err == nil {
        pro.PriceFloat, err = strconv.ParseFloat(pro.Price, 64)
        if err != nil { fmt.Println(err) }
        fmt.Printf("%+v\n", pro)
    } else {
        fmt.Println(err)
        fmt.Printf("%+v\n", pro)
    }
}
4
Mostafa

Évitez de convertir une chaîne en [] octet: b := []byte(s). Il alloue un nouvel espace mémoire et y copie l'intégralité du contenu.

strings.NewReader l'interface est meilleure. Voici le code de godoc:

package main

import (
    "encoding/json"
    "fmt"
    "io"
    "log"
    "strings"
)

func main() {
    const jsonStream = `
    {"Name": "Ed", "Text": "Knock knock."}
    {"Name": "Sam", "Text": "Who's there?"}
    {"Name": "Ed", "Text": "Go fmt."}
    {"Name": "Sam", "Text": "Go fmt who?"}
    {"Name": "Ed", "Text": "Go fmt yourself!"}
`
    type Message struct {
        Name, Text string
    }
    dec := json.NewDecoder(strings.NewReader(jsonStream))
    for {
        var m Message
        if err := dec.Decode(&m); err == io.EOF {
            break
        } else if err != nil {
            log.Fatal(err)
        }
        fmt.Printf("%s: %s\n", m.Name, m.Text)
    }
}
0
g10guang