我正在学习Python,目前正在学习更多的词典。
我在想;
如果我有一个字典,如:d = {'key_1': 'value_a', 'key_2': 'value_b'}
,我想将这个字典分开/分成变量,其中每个变量是字典中的一个键,每个变量值是字典中该键的值。
实现这一目标的最佳pythonic方法是什么?
d = {'key_1': 'value_a', 'key_2': 'value_b'}
#perform the command and get
key_1 = 'value_a'
key_2 = 'value_b'
我试过:key_1, key_2 = d
但它没有用。
基本上我正在寻求专家的智慧,以找出是否有更好的方法将2行代码减少为一行。
注意:这不是动态变量创建。
问题是dicts是无序的,所以你不能使用简单的d.values()
解包。您当然可以先按键对dict进行排序,然后解压缩值:
# Note: in python 3, items() functions as iteritems() did
# in older versions of Python; use it instead
ds = sorted(d.iteritems())
name0, name1, name2..., namen = [v[1] for v in ds]
你也可以,至少在一个对象中,做一些像:
for k, v in dict.iteritems():
setattr(self, k, v)
另外,正如我在上面的评论中提到的,如果您可以将需要解压缩字典的所有逻辑作为函数的变量,您可以执行以下操作:
def func(**kwargs):
# Do stuff with labeled args
func(**d)
以前没有提到过的解决方案
dictget = lambda d, *k: [d[i] for i in k]
然后使用它:
key_1, key_2 = dictget(d, 'key_1', 'key_2')
它的优点是即使需要检索更多变量,它仍然具有可读性。
然而,更具可读性的是“真实”功能,例如
def dictget(d, *k):
"""Get the values corresponding to the given keys in the provided dict."""
return [d[i] for i in k]
# or maybe
return (d[i] for i in k) # if we suppose that we have bigger sets of result
# or, equivalent to this
for i in k:
yield d[i]
它也支持使用docstring进行注释,并且是首选。
现有的答案可行,但它们基本上都是重新实现Python标准库中已存在的函数:operator.itemgetter()
来自docs:
返回一个可调用对象,该对象使用操作数的__getitem __()方法从其操作数中获取项。如果指定了多个项,则返回一个查找值元组。例如:
在f = itemgetter(2)之后,调用f(r)返回r [2]。
在g = itemgetter(2,5,3)之后,调用g(r)返回(r [2],r [5],r [3])。
换句话说,你的结构化dict赋值就像:
from operator import itemgetter
d = {'key_1': 'value_a', 'key_2': 'value_b'}
key_1, key_2 = itemgetter('key_1', 'key_2')(d)
# prints "Key 1: value_a, Key 2: value_b"
print("Key 1: {}, Key 2: {}".format(key_1, key_2))
我实际上有一个用例,我将__init__
方法的所有参数拉入对象构造的self命名空间:
vars(self).update(somedict)
vars函数为您提供与传递的对象关联的“命名空间”的字典。但是,由于CPython实现细节,这不适用于函数中的locals。所以它不应该适用于所有口译员。
对于全局命名空间,您可以将vars(self)
替换为globals()
,但这确实表明您的逻辑存在问题。如上所述,对于局部变量,这无论如何都不会起作用(即使你为dict赋值,它也会是NameError
)。
如果你勇敢的话,你可以做到这一点:
for k, v in d.items():
locals()[k] = v
但勇敢可能还不够 - 你可能也必须鲁莽等等。
如果你想成为像@ecatmur这样鲁莽的英雄,你可以这样做:
locals().update(d)
但是现在OP已经更新了他的问题并回答了评论,看来,这不是他真正想做的事情。仅供记录:动态创建变量有充分的理由 - 即使这里的每个人都认为它不是pythonic。通过动态更改范围,可以很好地解决许多解释器样式问题。只是以受控的方式做到这一点。并且......呃,不要将它部署到某个生产站点;)
我认为这应该可以解决你的问题
d = {'key_1': 'value_a', 'key_2': 'value_b'}
for k,v in d.items():
exec '%s=%s'%(k,v)
var1, var2 = (lambda key1, key2: (key1, key2))(**d)
如果你想让任何阅读你代码的人头疼,你可以使用匿名函数来解压缩这样的值。
不建议动态声明变量,因为找到变量名的来源变得非常困难。也就是说,有可能破解Dynamic variable name in python解决方案,但我不推荐它。使用普通的旧词典可能会更好。
这是一个解决方案,它在调用堆栈帧上使用Python的inspect模块来确定如何从提供的字典中提取正确的值。现在唯一的检查是确保每个输出变量都有可用的值,但如果需要,可以随意添加其他检查。
import inspect
def update(src):
result = []
cprev = inspect.currentframe().f_back
#print(cprev.f_code.co_names)
for v in cprev.f_code.co_names[1:]:
if src is cprev.f_locals[v]: continue
if v in src:
result.append(src[v])
else:
raise NameError(v + " has no source for update")
return result
用法如下:
src={'a': 1, 'b': 2, 'c': 3}
a,b,c = update(src)