能否将JSON加载到OrderedDict中?

466 投票
6 回答
175662 浏览
提问于 2025-04-16 22:46

好的,我可以在json.dump中使用OrderedDict。也就是说,OrderedDict可以作为输入来生成JSON格式的数据。

但是,它能作为输出吗?如果可以的话,怎么做呢?在我的情况下,我想把数据load进一个OrderedDict,这样我就能保持文件中键的顺序。

如果不行,有没有什么替代的方法?

6 个回答

51

好消息!从版本3.6开始,cPython的实现保留了字典的插入顺序(https://mail.python.org/pipermail/python-dev/2016-September/146327.html)。这意味着json库现在默认也会保留顺序。我们来看看python 3.5和3.6之间的行为差异。代码如下:

import json
data = json.loads('{"foo":1, "bar":2, "fiddle":{"bar":2, "foo":1}}')
print(json.dumps(data, indent=4))

在python 3.5中,结果的顺序是不可预测的:

{
    "fiddle": {
        "bar": 2,
        "foo": 1
    },
    "bar": 2,
    "foo": 1
}

而在python 3.6的cPython实现中:

{
    "foo": 1,
    "bar": 2,
    "fiddle": {
        "bar": 2,
        "foo": 1
    }
}

更棒的消息是,从python 3.7开始,这已经成为语言的规范(而不仅仅是cPython 3.6+的实现细节):https://mail.python.org/pipermail/python-dev/2017-December/151283.html

所以你现在的问题的答案就是:升级到python 3.6吧! :)

129

适用于 Python 2.7 及以上版本的简单版本

my_ordered_dict = json.loads(json_str, object_pairs_hook=collections.OrderedDict)

或者适用于 Python 2.4 到 2.6 的版本

import simplejson as json
import ordereddict

my_ordered_dict = json.loads(json_str, object_pairs_hook=ordereddict.OrderedDict)
670

是的,你可以这么做。只需要在使用 JSONDecoder 的时候指定一个叫 object_pairs_hook 的参数。实际上,这个例子就是文档中给出的。

>>> json.JSONDecoder(object_pairs_hook=collections.OrderedDict).decode('{"foo":1, "bar": 2}')
OrderedDict([('foo', 1), ('bar', 2)])
>>> 

如果你不需要 Decoder 实例来做其他事情,可以像下面这样把这个参数传给 json.loads

>>> import json
>>> from collections import OrderedDict
>>> data = json.loads('{"foo":1, "bar": 2}', object_pairs_hook=OrderedDict)
>>> print json.dumps(data, indent=4)
{
    "foo": 1,
    "bar": 2
}
>>> 

使用 json.load 的方法也是一样的:

>>> data = json.load(open('config.json'), object_pairs_hook=OrderedDict)

撰写回答