detect duplicate in JSON String Golang

后端 未结 2 1217
孤城傲影
孤城傲影 2021-01-14 10:20

I have JSON string like

\"{\\\"a\\\": \\\"b\\\", \\\"a\\\":true,\\\"c\\\":[\\\"field_3 string 1\\\",\\\"field3 string2\\\"]}\"

how to dete

2条回答
  •  日久生厌
    2021-01-14 10:55

    Use the json.Decoder to walk through the JSON. When an object is found, walk through keys and values checking for duplicate keys.

    func check(d *json.Decoder, path []string) error {
        // Get next token from JSON
        t, err := d.Token()
        if err != nil {
            return err
        }
    
        delim, ok := t.(json.Delim)
    
        // There's nothing to do for simple values (strings, numbers, bool, nil)
        if !ok {
            return nil
        }
    
        switch delim {
        case '{':
            keys := make(map[string]bool)
            for d.More() {
                // Get field key
                t, err := d.Token()
                if err != nil {
                    return err
                }
                key := t.(string)
    
                // Check for duplicates
                if keys[key] {
                    fmt.Printf("Duplicate %s\n", strings.Join(append(path, key), "/"))
                }
                keys[key] = true
    
                // Check value
                if err := check(d, append(path, key)); err != nil {
                    return err
                }
            }
            // Consume trailing }
            if _, err := d.Token(); err != nil {
                return err
            }
    
        case '[':
            i := 0
            for d.More() {
                if err := check(d, append(path, strconv.Itoa(i))); err != nil {
                    return err
                }
                i++
            }
            // Consume trailing ]
            if _, err := d.Token(); err != nil {
                return err
            }
    
        }
        return nil
    }
    

    Here's how to call it:

    data := `{"a": "b", "a":true,"c":["field_3 string 1","field3 string2"], "d": {"e": 1, "e": 2}}`
    if err := check(json.NewDecoder(strings.NewReader(data)), nil); err != nil {
        log.Fatal(err)
    }
    

    The output is:

    Duplicate a
    Duplicate d/e
    

    Run it on the Playground

提交回复
热议问题