我可以将JSON加载到OrderedDict吗?


427

好的,所以我可以在中使用OrderedDict json.dump。也就是说,OrderedDict可以用作JSON的输入。

但是可以用作输出吗?如果可以,怎么办?就我而言,我想load放入OrderedDict,以便可以将键的顺序保留在文件中。

如果没有,是否有某种解决方法?


从未尝试维护顺序,尽管我可以肯定它会很有用。
feathj 2011年

1
是的,在我的情况下,我弥合了不同语言和应用程序之间的鸿沟,并且JSON可以很好地工作。但是键的顺序有点问题。如果能够简单地打勾json.load以使用OrderedDicts而不是Python中的Dicts,那就太好了。
c00kiemonster 2011年

3
JSON规范将对象类型定义为具有无序键...期望特定的键顺序是一个错误
Anentropic

3
按键排序通常不适用于任何功能要求。它主要是为了人类可读性。如果我只希望JSON打印精美,那么我不希望任何文档顺序发生任何变化。
泡菜

5
它还有助于避免较大的git差异!
理查德·拉斯特

Answers:


609

是的你可以。通过指定JSONDecoderobject_pairs_hook参数。实际上,这是文档中给出的确切示例。

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

您可以将此参数传递给json.loads(如果不需要出于其他目的的Decoder实例),如下所示:

>>> 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)

3
我很困惑。文档说object_pairs_hook被解码成对的每个文字调用。为什么不为JSON中的每个记录创建一个新的OrderedDict?
Tim Keating 2014年

3
嗯...文档有些措辞含糊。它们的意思是“将所有对解码的整个结果”将作为一个列表传递给object_pairs_hook,而不是“每个对将传递给object_pairs_hook”,
SingleNegationElimination14

但是会丢失输入json的原始顺序吗?
SIslam

惊讶地发现json.load默认情况下不保持其顺序,但看起来它仅是镜像json本身的功能- {}处于无序状态,但[]json中的此处
cardamom

1
@RandomCertainty是的,每次在解析源时遇到JSON对象时,OrderedDict都会使用该JSON 建立最终的python值。
SingleNegationElimination

125

适用于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)

4
啊,但是它不包含object_pairs_hook,这就是为什么在2.6中仍然需要simplejson的原因。;)
mjhm 2011年

8
要注意的是simplejsonordereddict它们是需要安装的独立库。
phunehehe 2011年

2
适用于python 2.7+:代码中为“ import json,collections”,适用于python2.6-系统中的“ aptitude install python-pip”和“ pip installordereddict”
ZiTAL 2012年

这比以前使用JSONDecoder的方法更加容易和快捷。
Natim 2012年

奇怪的是,在pypy中,包含的json会失败loads('{}', object_pairs_hook=OrderedDict)
Matthew Schinckel,

37

一些好消息!从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))

在py3.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!:)


1
尽管在给定的示例中我看到的行为与您相同,但是在Python 3.6.4的CPython实现中,对我来说json.loads('{"2": 2, "1": 1}')变得如此{'1': 1, '2': 2}
fuglede

1
@fuglede看起来像是对dict.__repr__键进行排序,同时保留了基本顺序。换句话说,json.loads('{"2": 2, "1": 1}').items()就是dict_items([('2', 2), ('1', 1)])即使repr(json.loads('{"2": 2, "1": 1}'))"{'1': 1, '2': 2}"
西蒙·夏雷特

@SimonCharette嗯,可能是;我实际上无法在conda的pkgs / main / win-64 :: python-3.6.4-h0c2934d_3中重现我自己的观察结果,因此这很难测试。
fuglede

但是,这实际上并没有多大帮助,因为“重新命名”键仍会破坏键的顺序。
Hubro

7

除了转储字典,您总是可以写出密钥列表,然后OrderedDict通过遍历列表来重建密钥?


1
+1为低技术解决方案。我已经在处理YAML的同一问题时做到了这一点,但是必须重复是有点la脚,特别是当基础格式保留顺序时。避免丢失dict中但从键列表中丢失的键/值对可能也是有意义的,将它们附加在所有显式排序的项之后。
Mu Mind '02

2
这个技术含量低的解决方案还保留了上下文,否则上下文并不一定要以导出的格式保存(IOW;有人看到了JSON,并且没有任何内容明确说明“如果对它们进行操作,这些键应保持此顺序”)。
2012年

是什么确定“转储”的键列表顺序正确?嵌套字典怎么办?似乎转储既需要处理该问题,又需要使用OrdereDicts 递归完成重建。
martineau

5

除了在字典旁边转储键的有序列表之外,另一种具有显式优点的低技术解决方案是转储键-值对的(有序)列表ordered_dict.items()。加载很简单OrderedDict(<list of key-value pairs>)。尽管JSON没有这个概念(JSON字典没有顺序),但这仍然可以处理有序字典。

利用json以正确顺序转储OrderedDict 的事实确实很好。但是,通常必须将所有 JSON字典作为OrderedDict 读取(通过object_pairs_hook参数)是不必要的繁琐操作,也不一定有意义,因此显式转换必须排序的字典也是有意义的。


4

如果指定object_pairs_hook参数,则通常使用的load命令将起作用:

import json
from  collections import OrderedDict
with open('foo.json', 'r') as fp:
    metrics_types = json.load(fp, object_pairs_hook=OrderedDict)
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.