Unmarshaling อ็อบเจ็กต์ JSON ที่ซ้อนกัน


122

มีไม่กี่คำถามในหัวข้อแต่ไม่มีพวกเขาดูเหมือนจะครอบคลุมกรณีของฉันจึงฉันกำลังสร้างใหม่

ฉันมี JSON ดังต่อไปนี้:

{"foo":{ "bar": "1", "baz": "2" }, "more": "text"}

มีวิธีที่จะ unmarshal คุณสมบัติแถบที่ซ้อนกันและกำหนดให้กับคุณสมบัติของโครงสร้างโดยตรงโดยไม่ต้องสร้างโครงสร้างที่ซ้อนกันหรือไม่?

วิธีแก้ปัญหาที่ฉันกำลังใช้อยู่ตอนนี้มีดังต่อไปนี้:

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(ซึ่งเป็นวัตถุขนาดใหญ่) ยกเว้นองค์ประกอบเฉพาะบางอย่าง

แนวทางที่ถูกต้องในกรณีนี้คืออะไร? ฉันไม่ได้มองหาแฮ็คแปลก ๆ ดังนั้นถ้าเป็นวิธีที่จะไปฉันก็สบายดี

คำตอบ:


67

มีวิธีที่จะ unmarshal คุณสมบัติแถบที่ซ้อนกันและกำหนดให้กับคุณสมบัติของโครงสร้างโดยตรงโดยไม่ต้องสร้างโครงสร้างที่ซ้อนกันหรือไม่?

ไม่การเข้ารหัส / json ไม่สามารถทำเคล็ดลับด้วย "> some> deep> childnode" เช่นการเข้ารหัส / xml ทำได้ โครงสร้างที่ซ้อนกันเป็นวิธีที่จะไป


1
เหตุใดสิ่งนี้จึงแตกต่างจากการเข้ารหัส / xml
Caleb Hearth

1
@CalebThompson โครงสร้างของ XML และ JSON แตกต่างกันอย่างสิ้นเชิงแม้ว่ากรณีธรรมดาจะดูเหมือนกันก็ตาม เนื้อหาของแท็ก XML เป็นแบบ: (แผนผังที่เรียงลำดับของแท็กย่อยหรือข้อความ) และแผนผังแอตทริบิวต์ที่ไม่เรียงลำดับ JSON เป็นเหมือนโครงสร้าง Go ดังนั้นการแมป JSON กับโครงสร้างจึงง่ายกว่ามากเพียงแค่จำลองโครงสร้างหลัง JSON ของคุณ
Volker

ในกรณีของฉันโครงสร้างของ JSON ไม่ได้ถูกตัดสินจริง ๆ ดังนั้นฉันจึงสามารถสร้างโครงสร้างและเมื่อฉันแยกวิเคราะห์โดยใช้แม็พของอินเทอร์เฟซ [สตริง] {} ฉันกำลังมีปัญหาสำหรับองค์ประกอบที่ซ้อนกัน ทำอะไรได้บ้าง.?
viveksinghggits

แต่ทำไมเราไม่สามารถเปิดเผยโครงสร้างภายในโครงสร้างได้?
Vitaly Zdanevich

29

เช่นเดียวกับที่ Volker กล่าวถึงโครงสร้างที่ซ้อนกันเป็นหนทางที่จะไป แต่ถ้าคุณไม่ต้องการโครงสร้างที่ซ้อนกันจริงๆคุณสามารถแทนที่ฟังก์ชัน 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
ฉันได้รับ & {FooBar: 1 FooBaz: 2 More:} "ข้อความ" หายไป
Guy Segev

@GuySegev ฉันดำเนินการต่อและอัปเดตคำตอบเพื่อแก้ไขปัญหานั้น ขอบคุณที่ชี้ให้เห็น
rexposadas

22

นี่คือตัวอย่างของวิธียกเลิกการตอบสนอง JSON จากพร็อกซีเซิร์ฟเวอร์ Safebrowsing v4 API sbserver: 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)
}

2
ขอบคุณที่แสดงให้เห็นว่า json.Unmarshal สามารถ unmarshal ข้อมูล json ที่ซ้อนกันลึก ๆ ปัญหาของฉันคือฉันอ่าน JSON จากไฟล์และลงเอยด้วยการเว้นช่องว่างเป็นศูนย์ ดีใจที่คุณแบ่งปันสิ่งนี้!
Rohanthewiz

12

ใช่. ด้วยgjsonสิ่งที่คุณต้องทำตอนนี้คือ:

bar := gjson.Get(json, "foo.bar")

barอาจเป็นคุณสมบัติของโครงสร้างหากคุณต้องการ นอกจากนี้ยังไม่มีแผนที่


1
fastjsonยังอนุญาตให้ใช้เคล็ดลับเดียวกัน: fastjson.GetString(json, "foo", "bar")
valyala

9

แล้วช่องที่ไม่ระบุชื่อล่ะ? ฉันไม่แน่ใจว่าจะเป็นการ "โครงสร้างที่ซ้อนกัน" หรือไม่ แต่จะสะอาดกว่าการประกาศโครงสร้างแบบซ้อนกัน จะเป็นอย่างไรหากคุณต้องการนำองค์ประกอบที่ซ้อนกันกลับมาใช้ที่อื่น

type NestedElement struct{
    someNumber int `json:"number"`
    someString string `json:"string"`
}

type BaseElement struct {
    NestedElement `json:"bar"`
}

1

กำหนดค่าของซ้อนjsonให้กับ struct จนกว่าคุณจะทราบชนิดของคีย์ 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)
}

0

ฉันกำลังทำงานเกี่ยวกับสิ่งนี้ แต่ใช้งานได้กับโครงสร้างที่สร้างจากโปรโตเท่านั้น 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"
}

2
เพิ่มบรรทัดสองสามบรรทัดเพื่ออธิบายว่าสิ่งนี้ตอบคำถามอย่างไร หาก repo ถูกลบจะไม่มีค่าเหลืออยู่ในคำตอบ
Ubercool

ฉันไม่คิดว่าเขาจะกลับมาเพื่อน
DevX

-1

การรวมแผนที่และโครงสร้างทำให้อ็อบเจ็กต์ JSON ที่ซ้อนกันแบบ unmarshaling โดยที่คีย์เป็นไดนามิก => แผนที่ [สตริง]

ตัวอย่างเช่น 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)
}

คีย์แบบไดนามิกในแฮชจัดการสตริงและอ็อบเจ็กต์ที่ซ้อนกันจะแสดงด้วยโครงสร้าง


3
ดูเหมือนว่าจะไม่สมบูรณ์ ดิบไม่ได้ใช้
buildmaestro
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.