从文本文件中检索JSON对象(使用Python)

Lej*_*lek 19 python json object

我有成千上万个包含多个JSON对象的文本文件,但不幸的是,对象之间没有分隔符.对象存储为字典,其中一些字段本身就是对象.每个对象可能具有可变数量的嵌套对象.具体来说,对象可能如下所示:

{field1: {}, field2: "some value", field3: {}, ...} 
Run Code Online (Sandbox Code Playgroud)

并且在文本文件中没有分隔符的情况下连接数百个这样的对象.这意味着我既不能使用json.load()也不能json.loads().

关于如何解决这个问题的任何建议.有没有一个已知的解析器来做到这一点?

tba*_*ack 22

这会从字符串中解码JSON对象的"列表":

from json import JSONDecoder

def loads_invalid_obj_list(s):
    decoder = JSONDecoder()
    s_len = len(s)

    objs = []
    end = 0
    while end != s_len:
        obj, end = decoder.raw_decode(s, idx=end)
        objs.append(obj)

    return objs
Run Code Online (Sandbox Code Playgroud)

这里的好处是你可以和解析器一起玩.因此,它一直告诉你它找到错误的确切位置.

例子

>>> loads_invalid_obj_list('{}{}')
[{}, {}]

>>> loads_invalid_obj_list('{}{\n}{')
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "decode.py", line 9, in loads_invalid_obj_list
    obj, end = decoder.raw_decode(s, idx=end)
  File     "/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/json/decoder.py", line 376, in raw_decode
    obj, end = self.scan_once(s, idx)
ValueError: Expecting object: line 2 column 2 (char 5)
Run Code Online (Sandbox Code Playgroud)

清洁解决方案(稍后补充)

import json
import re

#shameless copy paste from json/decoder.py
FLAGS = re.VERBOSE | re.MULTILINE | re.DOTALL
WHITESPACE = re.compile(r'[ \t\n\r]*', FLAGS)

class ConcatJSONDecoder(json.JSONDecoder):
    def decode(self, s, _w=WHITESPACE.match):
        s_len = len(s)

        objs = []
        end = 0
        while end != s_len:
            obj, end = self.raw_decode(s, idx=_w(s, end).end())
            end = _w(s, end).end()
            objs.append(obj)
        return objs
Run Code Online (Sandbox Code Playgroud)

例子

>>> print json.loads('{}', cls=ConcatJSONDecoder)
[{}]

>>> print json.load(open('file'), cls=ConcatJSONDecoder)
[{}]

>>> print json.loads('{}{} {', cls=ConcatJSONDecoder)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/json/__init__.py", line 339, in loads
    return cls(encoding=encoding, **kw).decode(s)
  File "decode.py", line 15, in decode
    obj, end = self.raw_decode(s, idx=_w(s, end).end())
  File "/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/json/decoder.py", line 376, in raw_decode
    obj, end = self.scan_once(s, idx)
ValueError: Expecting object: line 1 column 5 (char 5)
Run Code Online (Sandbox Code Playgroud)