Unmarshaling вложенные объекты JSON в Голанге
Есть a несколько questions в теме но ни один из них, похоже, не охватывает мой случай, поэтому я создаю новый.
У меня JSON выглядит следующим образом:
{"foo":{ "bar": "1", "baz": "2" }, "more": "text"}
Есть ли способ развязать вложенное свойство bar и назначить его непосредственно свойству struct без создания вложенной структуры?
Решение, которое я сейчас принимаю, следующее:
type Foo struct {
More String `json:"more"`
Foo struct {
Bar string `json:"bar"`
Baz string `json:"baz"`
} `json:"foo"`
// FooBar string `json:"foo.bar"`
}
Это упрощенная версия, пожалуйста, игнорируйте многословие. Как вы можете видеть, я хотел бы проанализировать и присвоить значение
// FooBar string `json:"foo.bar"`
Я видел людей, использующих карту, но это не мое дело. Я в основном не забочусь о содержании foo
(который является крупным объектом), за исключением нескольких конкретных элементов.
Каков правильный подход в этом случае? Я не ищу странных хаков, поэтому, если это путь, я в порядке с этим.
Ответы
Ответ 1
Есть ли способ развязать вложенное свойство bar и назначить его непосредственно свойству struct без создания вложенной структуры?
Нет, кодирование /json не может выполнить трюк с помощью → some > deep > childnode ", как это делает encoding/xml.
Вложенные структуры - это путь.
Ответ 2
Как и упомянул Волкер, вложенные структуры - это путь. Но если вы действительно не хотите вложенных структур, вы можете переопределить функцию UnmarshalJSON.
https://play.golang.org/p/dqn5UdqFfJt
type A struct {
FooBar string // takes foo.bar
FooBaz string // takes foo.baz
More string
}
func (a *A) UnmarshalJSON(b []byte) error {
var f interface{}
json.Unmarshal(b, &f)
m := f.(map[string]interface{})
foomap := m["foo"]
v := foomap.(map[string]interface{})
a.FooBar = v["bar"].(string)
a.FooBaz = v["baz"].(string)
a.More = m["more"].(string)
return nil
}
Пожалуйста, игнорируйте тот факт, что я не возвращаю правильную ошибку. Я упустил это для простоты.
ОБНОВЛЕНИЕ: правильно получить значение "больше".
Ответ 3
Это пример того, как отменить ответ JSON с помощью прокси-сервера sbserver Safebrowsing v4 API: https://play.golang.org/p/4rGB5da0Lt
// this example shows how to unmarshall JSON requests from the Safebrowsing v4 sbserver
package main
import (
"fmt"
"log"
"encoding/json"
)
// response from sbserver POST request
type Results struct {
Matches []Match
}
// nested within sbserver response
type Match struct {
ThreatType string
PlatformType string
ThreatEntryType string
Threat struct {
URL string
}
}
func main() {
fmt.Println("Hello, playground")
// sample POST request
// curl -X POST -H 'Content-Type: application/json'
// -d '{"threatInfo": {"threatEntries": [{"url": "http://testsafebrowsing.appspot.com/apiv4/ANY_PLATFORM/MALWARE/URL/"}]}}'
// http://127.0.0.1:8080/v4/threatMatches:find
// sample JSON response
jsonResponse := `{"matches":[{"threatType":"MALWARE","platformType":"ANY_PLATFORM","threatEntryType":"URL","threat":{"url":"http://testsafebrowsing.appspot.com/apiv4/ANY_PLATFORM/MALWARE/URL/"}}]}`
res := &Results{}
err := json.Unmarshal([]byte(jsonResponse), res)
if(err!=nil) {
log.Fatal(err)
}
fmt.Printf("%v\n",res)
fmt.Printf("\tThreat Type: %s\n",res.Matches[0].ThreatType)
fmt.Printf("\tPlatform Type: %s\n",res.Matches[0].PlatformType)
fmt.Printf("\tThreat Entry Type: %s\n",res.Matches[0].ThreatEntryType)
fmt.Printf("\tURL: %s\n",res.Matches[0].Threat.URL)
}
Ответ 4
Да. С gjson все, что вам нужно сделать, это:
bar := gjson.Get(json, "foo.bar")
bar
может быть свойство struct, если хотите. Кроме того, нет карт.
Ответ 5
Как насчет анонимных полей? Я не уверен, что это будет "вложенная структура", но более чистая, чем наличие вложенной декларации структуры. Что делать, если вы хотите повторно использовать вложенный элемент в другом месте?
type NestedElement struct{
someNumber int `json:"number"`
someString string `json:"string"`
}
type BaseElement struct {
NestedElement `json:"bar"`
}
Ответ 6
Присвойте значения вложенному json
чтобы структурировать, пока не узнаете базовый тип ключей json:
package main
import (
"encoding/json"
"fmt"
)
// Object
type Object struct {
Foo map[string]map[string]string 'json:"foo"'
More string 'json:"more"'
}
func main(){
someJSONString := []byte('{"foo":{ "bar": "1", "baz": "2" }, "more": "text"}')
var obj Object
err := json.Unmarshal(someJSONString, &obj)
if err != nil{
fmt.Println(err)
}
fmt.Println("jsonObj", obj)
}
Ответ 7
Я работал над чем-то вроде этого. Но работает только со структурами, сгенерированными из прото. https://github.com/flowup-labs/grpc-utils
в вашем прото
message Msg {
Firstname string = 1 [(gogoproto.jsontag) = "name.firstname"];
PseudoFirstname string = 2 [(gogoproto.jsontag) = "lastname"];
EmbedMsg = 3 [(gogoproto.nullable) = false, (gogoproto.embed) = true];
Lastname string = 4 [(gogoproto.jsontag) = "name.lastname"];
Inside string = 5 [(gogoproto.jsontag) = "name.inside.a.b.c"];
}
message EmbedMsg{
Opt1 string = 1 [(gogoproto.jsontag) = "opt1"];
}
Тогда ваш вывод будет
{
"lastname": "Three",
"name": {
"firstname": "One",
"inside": {
"a": {
"b": {
"c": "goo"
}
}
},
"lastname": "Two"
},
"opt1": "var"
}
Ответ 8
Комбинирование map и struct позволяет демаршировать вложенные объекты JSON, где ключ является динамическим. => карта [строка]
Например: stock.json
{
"MU": {
"symbol": "MU",
"title": "micro semiconductor",
"share": 400,
"purchase_price": 60.5,
"target_price": 70
},
"LSCC":{
"symbol": "LSCC",
"title": "lattice semiconductor",
"share": 200,
"purchase_price": 20,
"target_price": 30
}
}
Перейти приложение
package main
import (
"encoding/json"
"fmt"
"io/ioutil"
"log"
"os"
)
type Stock struct {
Symbol string 'json:"symbol"'
Title string 'json:"title"'
Share int 'json:"share"'
PurchasePrice float64 'json:"purchase_price"'
TargetPrice float64 'json:"target_price"'
}
type Account map[string]Stock
func main() {
raw, err := ioutil.ReadFile("stock.json")
if err != nil {
fmt.Println(err.Error())
os.Exit(1)
}
var account Account
log.Println(account)
}
Динамический ключ в хэше - это дескриптор строки, а вложенный объект представлен структурой.