将字典划分为变量

问题描述 投票:19回答:9

我正在学习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行代码减少为一行。

注意:这不是动态变量创建。

python python-2.7
9个回答
14
投票

问题是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)

16
投票

以前没有提到过的解决方案

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进行注释,并且是首选。


15
投票

现有的答案可行,但它们基本上都是重新实现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))

2
投票

我实际上有一个用例,我将__init__方法的所有参数拉入对象构造的self命名空间:

vars(self).update(somedict)

vars函数为您提供与传递的对象关联的“命名空间”的字典。但是,由于CPython实现细节,这不适用于函数中的locals。所以它不应该适用于所有口译员。

对于全局命名空间,您可以将vars(self)替换为globals(),但这确实表明您的逻辑存在问题。如上所述,对于局部变量,这无论如何都不会起作用(即使你为dict赋值,它也会是NameError)。


2
投票

如果你勇敢的话,你可以做到这一点:

for k, v in d.items():
    locals()[k] = v

但勇敢可能还不够 - 你可能也必须鲁莽等等。

如果你想成为像@ecatmur这样鲁莽的英雄,你可以这样做:

locals().update(d)

但是现在OP已经更新了他的问题并回答了评论,看来,这不是他真正想做的事情。仅供记录:动态创建变量有充分的理由 - 即使这里的每个人都认为它不是pythonic。通过动态更改范围,可以很好地解决许多解释器样式问题。只是以受控的方式做到这一点。并且......呃,不要将它部署到某个生产站点;)


1
投票

我认为这应该可以解决你的问题

d = {'key_1': 'value_a', 'key_2': 'value_b'}
for k,v in d.items():
    exec '%s=%s'%(k,v)

1
投票
var1, var2 = (lambda key1, key2: (key1, key2))(**d)

如果你想让任何阅读你代码的人头疼,你可以使用匿名函数来解压缩这样的值。


0
投票

不建议动态声明变量,因为找到变量名的来源变得非常困难。也就是说,有可能破解Dynamic variable name in python解决方案,但我不推荐它。使用普通的旧词典可能会更好。


0
投票

这是一个解决方案,它在调用堆栈帧上使用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)
© www.soinside.com 2019 - 2024. All rights reserved.