将字典作为关键字参数传递给函数


344

我想使用字典在python中调用一个函数。

这是一些代码:

d = dict(param='test')

def f(param):
    print(param)

f(d)

这可以打印,{'param': 'test'}但我希望只打印test

我希望它可以类似地工作以获取更多参数:

d = dict(p1=1, p2=2)
def f2(p1, p2):
    print(p1, p2)
f2(d)

这可能吗?

Answers:


528

最后自己解决了。很简单,我只是缺少**运算符来解开字典

因此,我的示例变为:

d = dict(p1=1, p2=2)
def f2(p1,p2):
    print p1, p2
f2(**d)

57
如果您想借此帮助其他人,则应重新表述您的问题:问题不是通过字典,您想要的是将字典变成关键字参数
Javier

11
值得注意的是,您还可以将列表解压缩为位置参数:f2(* [1,2])
马修·特雷弗

10
“ dereference”:在此Python上下文中,通常的术语是“ unpack”。:)
mipadi

2
很好,只需将其与argparse / __ dict__结合使用,即可非常轻松地将命令行参数直接解析为类对象的选项。
Horus 2012年

1
将字典作为参数传递给函数时,我们要解压缩字典的原因是什么?
蒙娜·

128
In[1]: def myfunc(a=1, b=2):
In[2]:    print(a, b)

In[3]: mydict = {'a': 100, 'b': 200}

In[4]: myfunc(**mydict)
100 200

一些可能有助于了解的其他详细信息(阅读本文并经过测试后遇到的问题):

  1. 该函数可以具有字典中包含的参数
  2. 不能覆盖字典中已经存在的参数
  3. 字典不能包含函数中没有的参数。

例子:

数字1:该函数可以具有字典中未包含的参数

In[5]: mydict = {'a': 100}
In[6]: myfunc(**mydict)
100 2

数字2:您不能覆盖字典中已经存在的参数

In[7]: mydict = {'a': 100, 'b': 200}
In[8]: myfunc(a=3, **mydict)

TypeError: myfunc() got multiple values for keyword argument 'a'

数字3:字典中不能包含函数中没有的参数。

In[9]:  mydict = {'a': 100, 'b': 200, 'c': 300}
In[10]: myfunc(**mydict)

TypeError: myfunc() got an unexpected keyword argument 'c'

根据注释中的要求,第3号的解决方案是根据函数中可用的关键字参数来过滤字典:

In[11]: import inspect
In[12]: mydict = {'a': 100, 'b': 200, 'c': 300}
In[13]: filtered_mydict = {k: v for k, v in mydict.items() if k in [p.name for p in inspect.signature(myfunc).parameters.values()]}
In[14]: myfunc(**filtered_mydict)
100 200

另一种选择是在函数中接受(并忽略)其他kwarg:

In[15]: def myfunc2(a=None, **kwargs):
In[16]:    print(a)

In[17]: mydict = {'a': 100, 'b': 200, 'c': 300}

In[18]: myfunc2(**mydict)
100

请注意,除了可以有效地使用位置参数和列表或元组之外,还可以使用与kwargs相同的方式,这是一个更高级的示例,其中同时包含位置和关键字args:

In[19]: def myfunc3(a, *posargs, b=2, **kwargs):
In[20]:    print(a, b)
In[21]:    print(posargs)
In[22]:    print(kwargs)

In[23]: mylist = [10, 20, 30]
In[24]: mydict = {'b': 200, 'c': 300}

In[25]: myfunc3(*mylist, **mydict)
10 200
(20, 30)
{'c': 300}

4
对print.format使用解压缩特别有用。例如:'hello {greeting} {name}'.format( **{'name': 'Andrew', 'greeting': 'Mr'})
Martlark

旧问题,但仍然很相关。感谢您的详细回复。您是否知道解决案例3的任何方法?意思是当字典中的项目多于参数时,将字典中的项映射到函数参数吗?
spencer

2
@spencer解决方案已添加到答案中。
戴维·帕克斯

33

在python中,这称为“拆包”,您​​可以在本教程中找到有关它的信息。我同意,它的文档很烂,尤其是因为它是如此的有用。


20
最好将链接的相关内容复制到您的答案中,而不是依赖于生存到时间结束的链接。
理查德

3
@理查德(Richard)是关于网络的深刻哲学见解,对此我深表歉意!las,我在这里没有足够的空间来分享我的精彩证明...
llimllib 2014年

@llimllib,那我得问威尔斯博士!
理查德

6

在这里-可以进行其他任何迭代:

d = {'param' : 'test'}

def f(dictionary):
    for key in dictionary:
        print key

f(d)

似乎人们对此表示不满,因为它回答了最初的问题,而不是重新表达的问题。我建议立即删除此帖子。
dotancohen 2013年

@dotancohen不,它永远都不正确,它使始终伴随该问题的第二个代码块失败。从字面上看,印刷品就是一个例子。
戴夫·希利尔

它确实回答了这个问题,但它不是通过字典解包来完成的。根据发布的问题,他的方法是完全有效的。
Natecat '16
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.