python json模块的内部实现_python – 如何使用JSON模块进行漂亮打印时实现自定义缩进?...

所以我使用Python 2.7,使用json模块编码以下数据结构:

'layer1': {

'layer2': {

'layer3_1': [ long_list_of_stuff ],

'layer3_2': 'string'

}

}

我的问题是我使用漂亮的打印方式打印出来,如下所示:

json.dumps(data_structure, indent=2)

哪个很好,除了我想缩进所有内容,除了“layer3_1”中的内容 – 它是一个列出坐标的大量字典,因此,每个上面都设置了一个值,这使得漂亮的打印创建了一个包含数千行的文件,举例如下:

{

"layer1": {

"layer2": {

"layer3_1": [

{

"x": 1,

"y": 7

},

{

"x": 0,

"y": 4

},

{

"x": 5,

"y": 3

},

{

"x": 6,

"y": 9

}

],

"layer3_2": "string"

}

}

}

我真正想要的是类似于以下内容:

{

"layer1": {

"layer2": {

"layer3_1": [{"x":1,"y":7},{"x":0,"y":4},{"x":5,"y":3},{"x":6,"y":9}],

"layer3_2": "string"

}

}

}

我听说可以扩展json模块:是否可以将其设置为仅在“layer3_1”对象内部时关闭缩进?如果是这样,有人请告诉我怎么样?

解决方法:

更新

以下是我原来答案的一个版本,已经多次修改过.不同于原版,我发布的只是为了展示如何在J.F.Sebastian的answer中获得第一个想法,并且像他一样,返回了对象的非缩进字符串表示.最新更新版本返回单独格式化的Python对象JSON.

按照OP的注释之一,每个坐标dict的键将按排序顺序出现,但只有在驱动进程的初始json.dumps()调用中指定了sort_keys = True关键字参数,并且它不再更改对象的类型为沿途的字符串.换句话说,现在保持“包裹”对象的实际类型.

我认为不理解我的帖子的原始意图导致一些人低估它 – 所以,主要是因为这个原因,我已经“修复”并多次改进我的答案.当前版本是我原来的答案与@Erik Allik在他的answer中使用的一些想法的混合,以及来自其他用户的有用反馈,显示在此答案的下方评论中.

以下代码在Python 2.7.16和3.7.4中似乎都没有改变.

from _ctypes import PyObj_FromPtr

import json

import re

class NoIndent(object):

""" Value wrapper. """

def __init__(self, value):

self.value = value

class MyEncoder(json.JSONEncoder):

FORMAT_SPEC = '@@{}@@'

regex = re.compile(FORMAT_SPEC.format(r'(\d+)'))

def __init__(self, **kwargs):

# Save copy of any keyword argument values needed for use here.

self.__sort_keys = kwargs.get('sort_keys', None)

super(MyEncoder, self).__init__(**kwargs)

def default(self, obj):

return (self.FORMAT_SPEC.format(id(obj)) if isinstance(obj, NoIndent)

else super(MyEncoder, self).default(obj))

def encode(self, obj):

format_spec = self.FORMAT_SPEC # Local var to expedite access.

json_repr = super(MyEncoder, self).encode(obj) # Default JSON.

# Replace any marked-up object ids in the JSON repr with the

# value returned from the json.dumps() of the corresponding

# wrapped Python object.

for match in self.regex.finditer(json_repr):

# see https://stackoverflow.com/a/15012814/355230

id = int(match.group(1))

no_indent = PyObj_FromPtr(id)

json_obj_repr = json.dumps(no_indent.value, sort_keys=self.__sort_keys)

# Replace the matched id string with json formatted representation

# of the corresponding Python object.

json_repr = json_repr.replace(

'"{}"'.format(format_spec.format(id)), json_obj_repr)

return json_repr

if __name__ == '__main__':

from string import ascii_lowercase as letters

data_structure = {

'layer1': {

'layer2': {

'layer3_1': NoIndent([{"x":1,"y":7}, {"x":0,"y":4}, {"x":5,"y":3},

{"x":6,"y":9},

{k: v for v, k in enumerate(letters)}]),

'layer3_2': 'string',

'layer3_3': NoIndent([{"x":2,"y":8,"z":3}, {"x":1,"y":5,"z":4},

{"x":6,"y":9,"z":8}]),

'layer3_4': NoIndent(list(range(20))),

}

}

}

print(json.dumps(data_structure, cls=MyEncoder, sort_keys=True, indent=2))

输出:

{

"layer1": {

"layer2": {

"layer3_1": [{"x": 1, "y": 7}, {"x": 0, "y": 4}, {"x": 5, "y": 3}, {"x": 6, "y": 9}, {"a": 0, "b": 1, "c": 2, "d": 3, "e": 4, "f": 5, "g": 6, "h": 7, "i": 8, "j": 9, "k": 10, "l": 11, "m": 12, "n": 13, "o": 14, "p": 15, "q": 16, "r": 17, "s": 18, "t": 19, "u": 20, "v": 21, "w": 22, "x": 23, "y": 24, "z": 25}],

"layer3_2": "string",

"layer3_3": [{"x": 2, "y": 8, "z": 3}, {"x": 1, "y": 5, "z": 4}, {"x": 6, "y": 9, "z": 8}],

"layer3_4": [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19]

}

}

}

标签:python,json,indentation

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值