Lendo JSON de um arquivo?

320

Estou com um pouco de dor de cabeça só porque uma declaração simples e fácil está lançando alguns erros na minha cara.

Eu tenho um arquivo json chamado strings.json assim:

"strings": [{"-name": "city", "#text": "City"}, {"-name": "phone", "#text": "Phone"}, ...,
            {"-name": "address", "#text": "Address"}]

Eu quero ler o arquivo json, apenas isso por enquanto. Eu tenho estas declarações que eu descobri, mas não está funcionando:

import json
from pprint import pprint

with open('strings.json') as json_data:
    d = json.loads(json_data)
    json_data.close()
    pprint(d)

O erro exibido no console foi o seguinte:

Traceback (most recent call last):
  File "/home/.../android/values/manipulate_json.py", line 5, in <module>
    d = json.loads(json_data)
  File "/usr/lib/python2.7/json/__init__.py", line 326, in loads
    return _default_decoder.decode(s)
  File "/usr/lib/python2.7/json/decoder.py", line 366, in decode
    obj, end = self.raw_decode(s, idx=_w(s, 0).end())
TypeError: expected string or buffer
[Finished in 0.1s with exit code 1]

Editado

Alterado de json.loadsparajson.load

e entendi:

Traceback (most recent call last):
  File "/home/.../android/values/manipulate_json.py", line 5, in <module>
    d = json.load(json_data)
  File "/usr/lib/python2.7/json/__init__.py", line 278, in load
    **kw)
  File "/usr/lib/python2.7/json/__init__.py", line 326, in loads
    return _default_decoder.decode(s)
  File "/usr/lib/python2.7/json/decoder.py", line 369, in decode
    raise ValueError(errmsg("Extra data", s, end, len(s)))
ValueError: Extra data: line 829 column 1 - line 829 column 2 (char 18476 - 18477)
[Finished in 0.1s with exit code 1]
RRC
fonte
6
Você tem certeza de que o arquivo contém JSON válido?
Comprimidos de explosão
1
possível duplicata de valores Analisando de um arquivo JSON em Python
Pureferret
Veja também: Exemplo de leitura e gravação para JSON
Martin Thoma
Seu arquivo é um formato json inválido. Altere-o para: {"strings": [{"-name": "city", "#text": "City"}, {"-name": "phone", "#text": "Phone"}, ..., {"-name": "address", "#text": "Address"}]}
krizex 27/11

Respostas:

546

O json.load()método (sem "s" em "carga") pode ler um arquivo diretamente:

import json

with open('strings.json') as f:
    d = json.load(f)
    print(d)

Você estava usando o json.loads()método , usado apenas para argumentos de cadeia .

Edit: A nova mensagem é um problema totalmente diferente. Nesse caso, há algum json inválido nesse arquivo. Para isso, eu recomendaria executar o arquivo através de um validador json .

Também existem soluções para corrigir json, como por exemplo Como corrigir automaticamente uma sequência JSON inválida? .

ubomb
fonte
2
hm ... mudei de json.loads para json.load, mas recebo essa boa mensagem.
RRC
5
Ah, bem, a nova mensagem é um problema totalmente diferente. Nesse caso, há algum json inválido nesse arquivo. Para isso, eu recomendaria executar o arquivo através de um validador json .
ubomb
3
Entendi! Faltava o arquivo EOF. O arquivo não foi finalizado corretamente. Eu não notaria isso se não fosse sua boa recomendação! Obrigado!
RRC
1
ubomb, se você puder mudar, responda a mim para marcá-lo como aceito. Seja livre! Eu vou marcar.
RRC
Devo abrir o arquivo com sinalizador de bytes, para poder usar o método json.load, por quê? Eu tenho Py3.6
Grzegorz Krug
113

Aqui está uma cópia do código que funciona bem para mim

import json

with open("test.json") as json_file:
    json_data = json.load(json_file)
    print(json_data)

com os dados

{
    "a": [1,3,"asdf",true],
    "b": {
        "Hello": "world"
    }
}

convém agrupar sua linha json.load com uma captura de tentativa, porque JSON inválido causará uma mensagem de erro de rastreamento de pilha.

user1876508
fonte
41

O problema está usando com a instrução:

with open('strings.json') as json_data:
    d = json.load(json_data)
    pprint(d)

O arquivo já estará implicitamente fechado. Não há necessidade de ligar json_data.close()novamente.

Zongjun
fonte
1
Por favor, remova o json_data.close (). Como mencionado, será chamado implicitamente.
Bonnie Varghese
1
@Zongjun: Corrija as cargas em json.load (json_data).
Knight71
2
para imprimir bem, eu tive que usar: #print(json.dumps(d,sort_keys=True,indent=2))
Mike D
25

No python 3, podemos usar o método abaixo.

Leia do arquivo e converta para JSON

import json
from pprint import pprint

# Considering "json_list.json" is a json file

with open('json_list.json') as fd:
     json_data = json.load(fd)
     pprint(json_data)

com a instrução fecha automaticamente o descritor de arquivo aberto.


String para JSON

import json
from pprint import pprint

json_data = json.loads('{"name" : "myName", "age":24}')
pprint(json_data)
Thejesh PR
fonte
1

Você pode usar a biblioteca do pandas para ler o arquivo JSON.

import pandas as pd
df = pd.read_json('strings.json',lines=True)
print(df)
drorhun
fonte
0

Isso funciona para mim.

json.load () aceita o objeto file, analisa os dados JSON, preenche um dicionário Python com os dados e os retorna de volta para você.

Suponha que o arquivo JSON seja assim:

{
   "emp_details":[
                 {
                "emp_name":"John",
                "emp_emailId":"[email protected]"  
                  },
                {
                 "emp_name":"Aditya",
                 "emp_emailId":"[email protected]"
                }
              ] 
}

import json 

# Opening JSON file 
f = open('data.json',) 

# returns JSON object as  
# a dictionary 
data = json.load(f) 

# Iterating through the json 
# list 
for i in data['emp_details']: 
    print(i) 

# Closing file 
f.close()

#Output:
{'emp_name':'John','emp_emailId':'[email protected]'}
{'emp_name':'Aditya','emp_emailId':'[email protected]'}
Aditya patil
fonte