在 MongoDB 查询 JSON 数据遇到的 bug
问题描述
首先,假设有这么一个表:
key | value |
---|---|
password | "123456" |
view | 1000 |
object | [{name:"A", type: 1}] |
这是一个用于存储一些变量的表,由于变量类型多种多样,因此这里的 value 可能是任意的类型。这里问题就出现在 key 为object
的数据
在 Go 中,如果要查询这个表,最简单的办法是构造一个结构体
type V struct { Key string `bson:"key"` Value interface{} `bson:"value"` }
用这个结构体申请一个[]V
的变量,并用其接收查询结果
按照正常思路,这里应该接收到如下的数据:
[]V{ V{Key: "password", Value: "123456"}, V{Key: "view", Value: 1000}, V{Key: "object", Value: []map[string]interface{}{ map[string]interface{}{"name":"A", "type": 1}, }}, }
但是实际上,这里 object 获得是一个primitive.A
类型,并且每一个元素都是一个primitive.E
:
type E struct { Key string Value interface{} }
因此如果直接以这种形式查询,得到的结果是有问题的,毕竟 JSON 在 Go 中对应的类型应该是 map[string]interface{}
猜测原因
在 Go 的 MongoDB 驱动器中,针对几种不同类型的键值对,其实是有不同含义的,比如常见的map[string]interface{}
,由于其无序性,可能实现的结果和预期不一样,比如一个{name: "OhYee", age: 1}
的 JSON 数据,如果以 map
的形式传入 Go 中,可能会是 {age: 1, name: "OhYee"}
,因为age
的字典序小于name
。
当然,原则上来说这里本身就应该是无序的,但是由于各种原因,还是引入了primitive.D
这一类型,使用列表的有序性,将每一个键值对存储为primitive.E
中,并用列表保持顺序
由于在 Go 的驱动器中,存在两种不同的存储形式:有序字典、无序字典,而我们理解的 JSON 对应的应该是无序字典。当查询出有序字典时,就会出现上述问题
解决办法
如果不使用上面的结构体V
来接收数据,而是使用[]map[string]interface{}
定义的变量接收,则可以以无序字典形式获取数据(当然,顺序也没了)
在查询出结果后,再手动将其转换为结构 V
即可