16

我有一个变量名列表,如下所示:

['foo', 'bar', 'baz']

(我最初问我如何转换变量列表。请参阅下面的 Greg Hewgill 的回答。)

如何将其转换为字典,其中键是变量名(作为字符串),值是变量的值?

{'foo': foo, 'bar': bar, 'baz': baz}

现在我重新提出这个问题,我想出了:

d = {}
for name in list_of_variable_names:
    d[name] = eval(name)

可以改进吗?

更新,回答我为什么要这样做的问题(在评论中):

我经常发现自己使用 % 运算符对带有名称和值字典的字符串进行插值。通常字符串中的名称只是局部变量的名称。所以(下面的答案)我可以做这样的事情:

message = '''Name: %(name)s
ZIP: %(zip)s

Dear %(name)s,
...''' % dict((x, locals()[x]) for x in ['name', 'zip'])
4

4 回答 4

16

忘记过滤locals()!您提供给格式化字符串的字典允许包含未使用的键:

>>> name = 'foo'
>>> zip = 123
>>> unused = 'whoops!'
>>> locals()
{'name': 'foo', 'zip': 123, ... 'unused': 'whoops!', ...}
>>> '%(name)s %(zip)i' % locals()
'foo 123'

使用Python 3.6 中的新f-string 功能locals(),不再需要 using:

>>> name = 'foo'
>>> zip = 123
>>> unused = 'whoops!'
>>> f'{zip: >5} {name.upper()}'
'  123 FOO'
于 2008-10-23T20:29:33.377 回答
5

您可以使用列表或生成器推导来构建用于直接实例化 dict 的键、值元组的列表。最好的方法如下:

dict((name, eval(name)) for name in list_of_variable_names)

此外,例如,如果您知道变量存在于本地符号表中,您可以通过直接从本地查看变量来避免危险的 eval:

dict((name, locals()[name]) for name in list_of_variable_names)

在您最终更新之后,我认为下面的答案确实是您想要的。如果您只是将它用于使用您控制的字符串进行字符串扩展,只需将 locals() 直接传递给字符串扩展,它就会挑选出所需的值

但是,如果这些字符串可能来自外部源(例如翻译文件),那么过滤 locals() 是个好主意

于 2008-10-23T18:49:48.057 回答
4

您的原始列表[foo, bar, baz]不包含变量名称,它只包含引用与您列出的变量相同的值的元素。这是因为您可以有两个不同的变量名称来引用相同的值。

因此,列表本身不包含有关其他名称引用对象的信息。数组中的第一个元素具有名称foo,但它也具有名称a[0](假设您的数组称为a)。执行以下代码后,quux也引用同一个对象:

quux = a[0]

更新:您可以使用eval()它是对的,但通常不鼓励使用它。Python 提供了一个名为的特殊成员__dict__,其中包含当前模块的符号表。这样你就可以:

import __main__
d = dict((x, __main__.__dict__[x]) for x in list_of_variable_names)

import __main__当您的代码位于未命名的主模块中时,必须这样做是 Python 的一个怪癖。

于 2008-10-23T18:35:35.370 回答
1

效率不高,但没有调用eval

dict((k,v) for (k,v) in globals().iteritems() if k in list_of_variable_names)

或者

dict((k,v) for (k,v) in vars().iteritems() if k in list_of_variable_names)

取决于你想要什么。

于 2008-10-23T18:54:33.553 回答