Как декодировать JSON с преобразованием типа из строки в float64

86

Мне нужно декодировать строку JSON с числом с плавающей запятой, например:

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

Я использую код Голанга ниже:

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

Когда запускаю, получаю результат:

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

Я хочу знать, как декодировать строку JSON с помощью преобразования типа.

Янунон
источник

Ответы:

170

Ответ значительно менее сложен. Просто добавьте, чтобы сообщить интерпретатору JSON, что это строка с кодировкой float64 ,string(обратите внимание, что я изменил только 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)
    }
}
Дастин
источник
Спасибо! Думаю, это лучшее решение моей проблемы. Не могли бы вы сказать мне, где находится официальный документ об использовании ", string"?
yanunon 06
3
Обратите внимание, что запятая в начале json:",string"обязательна - без нее работать не будет.
Darrrrrren
1
@dustin есть идеи, как это сделать по-другому? Преобразовать номер json в строку? например, "N": 1234чтобы N: "1234"?
Камил Дзедзич
Это действительно здорово. За исключением того, что это не работает с ломтиками. Кто-нибудь знает, как это сделать для [] int? Простое добавление флага «, строка» дает: «Невозможно демаршалировать строку в значение int» ( play.golang.org/p/aFWSH4lUxv )
Далибор Филус 03
@KamilDziedzic, вы получили ответ на свой вопрос?
индивидуалист
12

Просто сообщаю вам, что вы можете сделать это без Unmarshalи использоватьjson.decode . Вот площадка для игр

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)
}
Сальвадор Дали
источник
4

Избегайте преобразования строки в [] байт: b := []byte(s). Он выделяет новое пространство памяти и копирует в него все содержимое.

strings.NewReaderинтерфейс лучше. Ниже приведен код от 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
источник
Ваш ответ подчеркивает важный момент, который я цитировал. Я что-то добавил? См. Stackoverflow.com/a/62740786/12817546
Tom L
3

Передача значения в кавычках делает его похожим на строку. + Изменить"price":"3460.00" на"price":3460.00 и все работает нормально.

Если вы не можете отбросить кавычки, вам придется разобрать его самостоятельно, используя 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)
    }
}
Мостафа
источник
Есть ли способ не изменять структуру «Продукт» и не реализовывать функцию или интерфейс декодирования для синтаксического анализа строки для плавания?
yanunon
1
@yanunon Да, вы можете использовать карту типа map[string]interface{}для Unmarshal, и разобрать , что в вашу структуру.
Мостафа,
@yanunon Или , если вы хотите реальную гибкость, вы можете написать свой собственный Unmarshal, который вызывает по умолчанию Unmarshalс map[string]interface{}, но использует reflectи strconvпакеты для выполнения синтаксического анализа.
Мостафа,