Як декодувати JSON за допомогою перетворення типу з рядка в float64 в Golang?


83

Мені потрібно декодувати рядок JSON з плаваючим номером, наприклад:

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

Я використовую код Golang нижче:

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 за допомогою перетворення типу.

Відповіді:


164

Відповідь значно менш складна. Просто додайте, скажіть інтерпетеру 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)
    }
}

Дякую! Я думаю, що це найкраще рішення для моєї проблеми. Не могли б ви сказати мені, де знаходиться офіційний документ про вживання ", рядок"?
Янунон

3
Зверніть увагу, що перед комою json:",string"потрібна кома перед цим - без цього вона не буде працювати.
Даррррррен,

1
@dustin будь-яка ідея, як це зробити навпаки? Перетворити номер JSON на рядок? наприклад "N": 1234до N: "1234"?
Каміль Дзедзіч

Це справді чудово. Окрім того, що це не працює на фрагментах. Хтось знає спосіб, як це зробити для [] int? Просто додаючи прапорець ", string" виходить: "неможливо зняти маршальський рядок у значення go int" ( play.golang.org/p/aFWSH4lUxv )
Далібор Філус 03.03.16

@KamilDziedzic чи отримали Ви рішення Вашого запиту?
Maverick

12

Просто повідомляючи вам, що ви можете зробити це без Unmarshalвикористання та використання json.decode. Ось ігровий майданчик Go

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

Передаючи значення в лапки, це виглядає як рядок. Перейдіть "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)
    }
}

Чи є спосіб, що не змінюючи структуру 'Product' та реалізуючи функцію декодування або інтерфейс для синтаксичного аналізу рядка для плавання?
yanunon

1
@yanunon Так, ви можете використовувати карту типу map[string]interface{}для Unmarshalта проаналізувати її у своїй структурі.
Мостафа

@yanunon Або якщо ви хочете отримати справжню гнучкість, ви можете написати свій власний Unmarshal, який за замовчуванням викликає Unmarshala map[string]interface{}, але використовує reflectта strconvпакети для аналізу.
Мостафа

3

Уникайте перетворення рядка в [] байт: 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)
    }
}

Ваша відповідь робить чудову думку, яку я цитував. Я додав до нього? Дивіться stackoverflow.com/a/62740786/12817546
Том Л
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.