17

私はjson apiを取得するためにループを実行しています.これが私のループにあるものです:

response_item = requests.request('GET',url_item,params=None,verify=False)
response_item = json.loads(response_item.text)
response_item = ast.literal_eval(json.dumps(response_item, ensure_ascii=False).encode('utf8'))

約 45000 個の json オブジェクトをスキャンし、反復ごとに「url_item」変数を生成します。各オブジェクトは同じです。7000 オブジェクトのようなものを取得でき、7064 番目に到達すると次のエラーが発生します。

Traceback (most recent call last):
  File "C:\Python27\tools\api_item.py", line 47, in <module>
    response_item = ast.literal_eval(json.dumps(response_item, ensure_ascii=False).encode('utf8'))
  File "C:\Python27\lib\ast.py", line 80, in literal_eval
    return _convert(node_or_string)
  File "C:\Python27\lib\ast.py", line 63, in _convert
    in zip(node.keys, node.values))
  File "C:\Python27\lib\ast.py", line 62, in <genexpr>
    return dict((_convert(k), _convert(v)) for k, v
  File "C:\Python27\lib\ast.py", line 63, in _convert
    in zip(node.keys, node.values))
  File "C:\Python27\lib\ast.py", line 62, in <genexpr>
    return dict((_convert(k), _convert(v)) for k, v
  File "C:\Python27\lib\ast.py", line 79, in _convert
    raise ValueError('malformed string')
ValueError: malformed string

以前は 2 番目と 3 番目の「response_item」を出力していました。もちろん、この場合、直前にエラーが発生したため、3 番目のエラーは表示されません。ここでは、json.load の後に出力するものを示します。

{u'restrictions': [], u'name': u'Sac \xe0 dos de base', u'level': 0, u'rarity': u'Basic', u'vendor_value': 11, u'details': {u'no_sell_or_sort': False, u'size': 20}, u'game_types': [u'Activity', u'Wvw', u'Dungeon', u'Pve'], u'flags': [u'NoSell', u'SoulbindOnAcquire', u'SoulBindOnUse'], u'icon': u'https://render.guildwars2.com/file/80E36806385691D4C0910817EF2A6C2006AEE353/61755.png', u'type': u'Bag', u'id': 8932, u'description': u'Un sac de 20 emplacements pour les personnages d\xe9butants.'}

これより前に取得したすべてのアイテムは同じタイプ、同じ形式であり、 7064 を除いてエラーはありません!

ご協力ありがとうございました!

4

2 に答える 2

43

JSON データでは使用しないでください。ast.literal_eval()JSON と Python のリテラルは同じように見えるかもしれませんが、実際にはそうではありません。

この場合、データには、falseJSON で設定されたブール値フラグが含まれています。適切な Python ブール値はタイトルケースを使用するため、次のようになりFalseます。

>>> import json, ast
>>> s = '{"no_sell_or_sort": false, "size": 20}'
>>> json.loads(s)
{u'no_sell_or_sort': False, u'size': 20}
>>> ast.literal_eval(s)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/Users/mj/Development/Library/buildout.python/parts/opt/lib/python2.7/ast.py", line 80, in literal_eval
    return _convert(node_or_string)
  File "/Users/mj/Development/Library/buildout.python/parts/opt/lib/python2.7/ast.py", line 63, in _convert
    in zip(node.keys, node.values))
  File "/Users/mj/Development/Library/buildout.python/parts/opt/lib/python2.7/ast.py", line 62, in <genexpr>
    return dict((_convert(k), _convert(v)) for k, v
  File "/Users/mj/Development/Library/buildout.python/parts/opt/lib/python2.7/ast.py", line 79, in _convert
    raise ValueError('malformed string')
ValueError: malformed string

その他の違いとしては、nullの代わりにを使用することや、Unicode エスケープ シーケンスを Python 2 にとってプレーンな (バイト) 文字列Noneのように見えるものに使用すること、非 BMP コードポイントをエスケープするときに UTF-16 サロゲートを使用することが含まれます。

json.loads()ではなくでデータをロードしますast.literal_eval()。適切な JSON を問題なく処理できるだけでなく、より高速です。

あなたの場合、使用しているように見えますので、 でjson.dumps()データを再度ロードしてみてくださいast.literal_eval()。その手順は必要ありません。既にPython オブジェクトが存在します。

つまり、次の行です。

response_item = ast.literal_eval(json.dumps(response_item, ensure_ascii=False).encode('utf8'))

せいぜい冗長であり、最悪の場合、非常に間違っています。JSON 文字列に再エンコードresponse_itemしても、Python リテラルとして解釈できるものは生成されません。

于 2015-09-21T13:03:39.113 に答える