我可以将JSON加载到OrderedDict中吗?

c00*_*ter 415 python json load ordereddictionary

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

但它可以用作输出吗?如果是这样的话?在我的情况下,我想load进入OrderedDict,这样我就可以保持文件中键的顺序.

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

Sin*_*ion 587

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

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

您可以将此参数传递给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
}
>>> 
Run Code Online (Sandbox Code Playgroud)

使用json.load方式相同:

>>> data = json.load(open('config.json'), object_pairs_hook=OrderedDict)
Run Code Online (Sandbox Code Playgroud)

  • 我很困惑.文档说,对每个被解码成对的文字调用object_pairs_hook.为什么不为JSON中的每条记录创建一个新的OrderedDict? (3认同)
  • 嗯......文档有点含糊不清.他们的意思是"解码所有对的整个结果"将按顺序传递给`object_pairs_hook`,而不是"每对将传递给object_pairs_hook", (3认同)

mjh*_*jhm 124

Python 2.7+的简单版本

my_ordered_dict = json.loads(json_str, object_pairs_hook=collections.OrderedDict)
Run Code Online (Sandbox Code Playgroud)

或者对于Python 2.4到2.6

import simplejson as json
import ordereddict

my_ordered_dict = json.loads(json_str, object_pairs_hook=ordereddict.OrderedDict)
Run Code Online (Sandbox Code Playgroud)

  • 要注意`simplejson`和`ordereddict`是您需要安装的独立库. (8认同)
  • 啊,但它不包括object_pairs_hook - 这就是为什么你仍然需要2.6中的simplejson.;) (4认同)
  • 对于python 2.7+:代码中的"import json,collections",用于系统中的python2.6-"aptitude install python-pip"和"pip install ordereddict" (2认同)

pel*_*son 34

一些好消息!从版本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))
Run Code Online (Sandbox Code Playgroud)

在py3.5中,结果顺序是未定义的:

{
    "fiddle": {
        "bar": 2,
        "foo": 1
    },
    "bar": 2,
    "foo": 1
}
Run Code Online (Sandbox Code Playgroud)

在python 3.6的cPython实现中:

{
    "foo": 1,
    "bar": 2,
    "fiddle": {
        "bar": 2,
        "foo": 1
    }
}
Run Code Online (Sandbox Code Playgroud)

真正的好消息是,这已成为python 3.7的语言规范(与cPython 3.6+的实现细节相反):https://mail.python.org/pipermail/python-dev/2017-December/151283 html的

所以你的问题的答案现在变成:升级到python 3.6!:)

  • @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}"`。 (2认同)

ntg*_*ntg 8

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

import json
from  collections import OrderedDict
with open('foo.json', 'r') as fp:
    metrics_types = json.load(fp, object_pairs_hook=OrderedDict)
Run Code Online (Sandbox Code Playgroud)


Amb*_*ber 7

除了转储dict之外,你总是可以写出密钥列表,然后OrderedDict通过遍历列表来重建它?

  • 低技术解决方案还保留了导出格式中未必保留的上下文(IOW;有人看到JSON,并且没有任何明确说明"如果这些键对它进行操作,这些键应该按此顺序保留"). (2认同)

Eri*_*got 5

除了将有序的键列表与字典一起转储之外,另一种具有显式优势的低技术解决方案是转储(有序)键值对列表ordered_dict.items(); 装载很简单OrderedDict(<list of key-value pairs>).尽管JSON没有这个概念(JSON字典没有顺序),但它处理有序字典.

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