json - 如何将 JSON 与不同的顺序进行比较?
问题描述
我正在尝试使用黄金文件实现测试,但是,我的函数生成的 JSON 顺序不同,但保持相同的值。我已经实现了这里使用的比较方法:
但这取决于顺序。正如布拉德在这里所说:
JSON 对象是无序的,就像 Go 地图一样。如果您依赖于特定实现序列化 JSON 对象的顺序,则存在错误。
我编写了一些示例代码来模拟我的困境:
package main
import (
"bufio"
"encoding/json"
"fmt"
"io/ioutil"
"math/rand"
"os"
"reflect"
"time"
)
type example struct {
Name string
Earnings float64
}
func main() {
slice := GetSlice()
gfile, err := ioutil.ReadFile("testdata/example.golden")
if err != nil {
fmt.Println(err)
fmt.Println("Failed reading golden file")
}
testJSON, err := json.Marshal(slice)
if err != nil {
fmt.Println(err)
fmt.Println("Error marshalling slice")
}
equal, err := JSONBytesEqual(gfile, testJSON)
if err != nil {
fmt.Println(err)
fmt.Println("Error comparing JSON")
}
if !equal {
fmt.Println("Restults don't match JSON")
} else {
fmt.Println("Success!")
}
}
func GetSlice() []example {
t := []example{
example{"Penny", 50.0},
example{"Sheldon", 70.0},
example{"Raj", 20.0},
example{"Bernadette", 200.0},
example{"Amy", 250.0},
example{"Howard", 1.0}}
rand.Seed(time.Now().UnixNano())
rand.Shuffle(len(t), func(i, j int) { t[i], t[j] = t[j], t[i] })
return t
}
func JSONBytesEqual(a, b []byte) (bool, error) {
var j, j2 interface{}
if err := json.Unmarshal(a, &j); err != nil {
return false, err
}
if err := json.Unmarshal(b, &j2); err != nil {
return false, err
}
return reflect.DeepEqual(j2, j), nil
}
func WriteTestSliceToFile(arr []example, filename string) {
file, err := os.OpenFile(filename, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
if err != nil {
fmt.Println("failed creating file: %s", err)
}
datawriter := bufio.NewWriter(file)
marshalledStruct, err := json.Marshal(arr)
if err != nil {
fmt.Println("Error marshalling json")
fmt.Println(err)
}
_, err = datawriter.Write(marshalledStruct)
if err != nil {
fmt.Println("Error writing to file")
fmt.Println(err)
}
datawriter.Flush()
file.Close()
}
解决方案
JSON数组是有序的。json.Marshal 函数在将切片编码为 JSON 数组时保留顺序。
JSON对象没有排序。json.Marshal 函数按照文档中描述的排序键顺序写入对象成员。
bradfitz 评论 JSON 对象排序与此问题无关:
- 问题中的应用程序正在使用 JSON 数组,而不是 JSON 对象。
- 在 Brad 发表评论几年后,该软件包已更新为按排序键顺序编写对象字段。
要在忽略顺序的情况下比较切片,请在比较之前对两个切片进行排序。这可以在编码为 JSON 之前或从 JSON 解码之后完成。
sort.Slice(slice, func(i, j int) bool {
if slice[i].Name != slice[j].Name {
return slice[i].Name < slice[j].Name
}
return slice[i].Earnings < slice[j].Earnings
})
推荐阅读
- javascript - 如何在jQuery中获取文件是否上传
- node.js - 一个模块中有很多路线.exports
- python - Python2 在使用 sorted 时会引发异常,但 python3 不会
- javascript - 使用jquery单击后退按钮后输入字段没有被禁用
- android - 程序类型已经存在:androidx.versionedparcelable.NonParcelField
- android - 类型不匹配。必需:暂停()→响应<*>找到:响应
- elasticsearch - Elasticsearch - 按日期字段按特定输入日期升序的脚本排序
- python - 是否可以找到或匹配两个具有不同特殊字符的名称 django
- header - 如何让 testcafe 复制和使用与实际网站相同的标题?
- c# - 单元测试以检查未使用的属性