web-dev-qa-db-ja.com

辞書を関数にキーワードパラメーターとして渡す

辞書を使用して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)

これは可能ですか?

295
Dave Hillier

最後に自分で考え出した。それは簡単です、辞書を展開するために**演算子が欠落していました

したがって、私の例は次のようになります。

d = dict(p1=1, p2=2)
def f2(p1,p2):
    print p1, p2
f2(**d)
456
Dave Hillier
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. 関数には、ディクショナリに含まれるnotのパラメーターを含めることができます
  2. ディクショナリに既に存在するパラメーターをnotオーバーライドできます
  3. 辞書にはnot関数にないパラメーターを含めることができます。

例:

数値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'

Number 3:辞書には、関数にないパラメーターを含めることはできません。

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

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

コメントで要求されているように、Number 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

別のオプションは、関数で追加のkwargsを受け入れる(および無視する)ことです。

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と実質的に同じ方法で位置引数とリストまたはタプルを使用できることよりも注意してください。上記のすべてを組み込んだより高度な例があります。

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}
90
David Parks

Pythonでは、これは「アンパック」と呼ばれ、 tutorial で少し見つけることができます。それのドキュメンテーションは、特に空想的に有用であるため、私は同意します。

30
llimllib

ここに行く-他の反復可能なものだけで動作します:

d = {'param' : 'test'}

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

f(d)
6