如何将字典的str表示,如下面的字符串,转换为字典?

s = "{'muffin' : 'lolz', 'foo' : 'kitty'}"

我不喜欢使用eval。我还能用什么?

主要原因是,我同事写的一个类,将所有输入转换为字符串。我没有心情去修改他的课程,去处理这个问题。


当前回答

使用json.loads:

>>> import json
>>> h = '{"foo":"bar", "foo2":"bar2"}'
>>> d = json.loads(h)
>>> d
{u'foo': u'bar', u'foo2': u'bar2'}
>>> type(d)
<type 'dict'>

其他回答

如果字符串始终是可信的,则可以使用eval(或建议使用literal_eval;不管字符串是什么,它都是安全的。)否则,您需要一个解析器。JSON解析器(例如simplejson)如果只存储符合JSON方案的内容,那么它就可以工作。

使用json。ast库消耗大量内存,而且速度较慢。我有一个进程,需要读取156Mb的文本文件。Ast有5分钟的延迟转换字典json和1分钟使用60%的内存!

以OP为例:

s = "{'muffin' : 'lolz', 'foo' : 'kitty'}"

我们可以使用Yaml来处理string中的这种非标准json:

>>> import yaml
>>> s = "{'muffin' : 'lolz', 'foo' : 'kitty'}"
>>> s
"{'muffin' : 'lolz', 'foo' : 'kitty'}"
>>> yaml.load(s)
{'muffin': 'lolz', 'foo': 'kitty'}

你可以使用内置的ast.literal_eval:

>>> import ast
>>> ast.literal_eval("{'muffin' : 'lolz', 'foo' : 'kitty'}")
{'muffin': 'lolz', 'foo': 'kitty'}

这比使用eval更安全。正如它自己的文件所说:

>>> help(ast.literal_eval)
Help on function literal_eval in module ast:

literal_eval(node_or_string)
    Safely evaluate an expression node or a string containing a Python
    expression.  The string or node provided may only consist of the following
    Python literal structures: strings, numbers, tuples, lists, dicts, booleans,
    and None.

例如:

>>> eval("shutil.rmtree('mongo')")
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<string>", line 1, in <module>
  File "/opt/Python-2.6.1/lib/python2.6/shutil.py", line 208, in rmtree
    onerror(os.listdir, path, sys.exc_info())
  File "/opt/Python-2.6.1/lib/python2.6/shutil.py", line 206, in rmtree
    names = os.listdir(path)
OSError: [Errno 2] No such file or directory: 'mongo'
>>> ast.literal_eval("shutil.rmtree('mongo')")
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/opt/Python-2.6.1/lib/python2.6/ast.py", line 68, in literal_eval
    return _convert(node_or_string)
  File "/opt/Python-2.6.1/lib/python2.6/ast.py", line 67, in _convert
    raise ValueError('malformed string')
ValueError: malformed string

https://docs.python.org/library/json.html

JSON可以解决这个问题,尽管它的解码器希望键和值周围有双引号。如果你不介意换一个黑客…

import json
s = "{'muffin' : 'lolz', 'foo' : 'kitty'}"
json_acceptable_string = s.replace("'", "\"")
d = json.loads(json_acceptable_string)
# d = {u'muffin': u'lolz', u'foo': u'kitty'}

注意,如果你有单引号作为你的键或值的一部分,这将失败,由于不适当的字符替换。只有当您非常厌恶eval解决方案时,才推荐使用此解决方案。

更多关于json的单引号:jQuery。由于转义了JSON中的单引号,parseJSON抛出“无效JSON”错误