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.
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)
}
}
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)
}
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)
}
}
É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)
}
}