Jak zdekodować JSON z typem Convert from string to float64

86

Muszę zdekodować ciąg JSON z liczbą zmiennoprzecinkową, taką jak:

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

Używam kodu Golang poniżej:

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

Kiedy go uruchomię, uzyskaj wynik:

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

Chcę wiedzieć, jak zdekodować ciąg JSON za pomocą konwersji typu.

yanunon
źródło

Odpowiedzi:

170

Odpowiedź jest znacznie mniej skomplikowana. Po prostu dodaj i powiedz interpeterowi JSON, że jest to kodowany ciągiem float64 z ,string(zauważ, że zmieniłem tylko Pricedefinicję):

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)
    }
}
Dustin
źródło
Dziękuję Ci! Myślę, że to najlepsze rozwiązanie mojego problemu. Czy możesz mi powiedzieć, gdzie jest oficjalny dokument dotyczący użycia „, string”?
yanunon
3
Pamiętaj, że przecinek na początku json:",string"jest konieczny - bez niego nie zadziała.
Darrrrrren
1
@dustin masz jakiś pomysł, jak to zrobić w inny sposób? Konwertować liczbę json, aby przejść do ciągu? np. "N": 1234do N: "1234"?
Kamil Dziedzic
To naprawdę niesamowite. Z wyjątkiem tego, że nie działa na plasterkach. Czy ktoś wie, jak to zrobić dla [] int? Po prostu dodanie flagi „, string” daje: „nie można unmarshal string do wartości go int” ( play.golang.org/p/aFWSH4lUxv )
Dalibor Filus
@KamilDziedzic czy dostałeś rozwiązanie swojego zapytania?
Maverick
12

Po prostu informuję, że możesz to zrobić bez Unmarshali używać json.decode. Oto 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)
}
Salvador Dali
źródło
4

Unikać konwertowania ciąg [] bajt: b := []byte(s). Alokuje nową przestrzeń pamięci i kopiuje do niej całą zawartość.

strings.NewReaderinterfejs jest lepszy. Poniżej znajduje się kod z 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)
    }
}
g10guang
źródło
Twoja odpowiedź jest świetnym argumentem, który zacytowałem. Czy dodałem do tego? Zobacz stackoverflow.com/a/62740786/12817546
Tom L
3

Przekazanie wartości w cudzysłowie sprawia, że ​​wygląda ona jak ciąg. Zmień "price":"3460.00"na "price":3460.00i wszystko działa dobrze.

Jeśli nie możesz upuścić cudzysłowów, musisz przeanalizować go samodzielnie, używając 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)
    }
}
Mostafa
źródło
Czy istnieje sposób, aby nie zmieniać struktury „produktu” i nie zaimplementować funkcji dekodującej lub interfejsu w celu przeanalizowania ciągu w celu uzyskania wartości float?
yanunon
1
@yanunon Tak, możesz użyć mapy typu map[string]interface{}dla Unmarshali przeanalizować ją w swojej strukturze.
Mostafa
@yanunon Lub jeśli chcesz prawdziwej elastyczności, możesz napisać własny Unmarshal, który wywołuje default Unmarshalz a map[string]interface{}, ale używa pakietów reflecti strconvdo wykonywania analizy.
Mostafa