ValueError:使用ast.literal_eval的格式错误的字符串

时间:2015-09-21 12:50:23

标签: python json string python-requests malformed

我正在做一个循环来获取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对象的东西,当我到达7064th时我有以下错误:

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

我曾经打印过第二个和第三个“response_item”。当然在这种情况下,第三个没有显示,因为我之前有错误,这里是我在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.'}

我在此之前获得的每个项目具有相同的类型,相同的格式,除了7064th之外我没有任何错误!

感谢您的帮助!

2 个答案:

答案 0 :(得分:22)

您应该在JSON数据上使用ast.literal_eval()。 JSON和Python文字可能看起来像同样的东西,但它们根本不是。

在这种情况下,您的数据包含一个布尔标志,在JSON中设置为false。一个合适的Python布尔值使用title-case,所以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而不是None,而使用UTF-16代理的Python 2中的Unicode转义序列看起来像 plain(字节)字符串当转义非BMP代码点时。

使用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'))

充其量是多余的,而且非常非常错,最坏的情况。将response_item重新编码为JSON字符串不会产生可以解释为Python文字的内容。

答案 1 :(得分:-3)

如果你使用它,

ast.literal_eval是安全的SQL注入。 因为当插入不需要的章程时,它将显示Syntex错误,这会阻止注入。