检测JSON String Golang中的重复项

Sum*_* Ji 2 json go

我有JSON字符串之类的

"{\"a\": \"b\", \"a\":true,\"c\":[\"field_3 string 1\",\"field3 string2\"]}"
Run Code Online (Sandbox Code Playgroud)

如何使用Golang检测此json字符串中的重复属性

Cer*_*món 7

使用json.Decoder来浏览JSON.找到对象时,遍历键和值检查重复键.

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
}
Run Code Online (Sandbox Code Playgroud)

以下是如何调用它:

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)
}
Run Code Online (Sandbox Code Playgroud)

输出是:

Duplicate a
Duplicate d/e
Run Code Online (Sandbox Code Playgroud)

在Playground上运行它

  • 很好的解决方案,但是您忘记在重复的情况下返回错误,因此 err 仍然为零。 (2认同)
  • @eyalP - 好点。最新的编辑解决了这个问题。 (2认同)

dav*_*ave 1

一种可能效果很好的方法是简单地解码、重新编码,然后对照旧 json 检查新 json 的长度:

https://play.golang.org/p/50P-x1fxCzp

package main

import (
  "encoding/json"
  "fmt"
)

func main() {
  jsn := []byte("{\"a\": \"b\", \"a\":true,\"c\":[\"field_3 string 1\",\"field3 string2\"]}")
  var m map[string]interface{}
  err := json.Unmarshal(jsn, &m)
  if err != nil {
      panic(err)
  }
        l := len(jsn)
        jsn, err = json.Marshal(m)
  if err != nil {
      panic(err)
  }
  if l != len(jsn) {
      panic(fmt.Sprintf("%s: %d (%d)", "duplicate key", l, len(jsn)))
  }
}
Run Code Online (Sandbox Code Playgroud)

正确的方法是重新实现该json.Decode函数,并存储找到的键映射,但上面的方法应该可以工作(特别是如果您首先使用 json 从 json 中删除任何空格jsn = bytes.Replace(jsn, []byte(" "), []byte(""), -1)以防止误报。

  • 你可能可以使用``json.Compact``来删除多余的空格 (2认同)
  • 我认为OP问的是“如何检测重复属性”,而不是“如何检测**是否存在**重复属性” (2认同)