Przekazywanie słownika do funkcji jako parametrów słów kluczowych

Chciałbym wywołać funkcję w Pythonie używając słownika.

Oto jakiś kod:

d = dict(param='test')

def f(param):
    print(param)

f(d)

To drukuje {'param': 'test'} ale chciałbym, żeby to po prostu wydrukowało test.

Chciałbym, aby działało podobnie dla większej liczby parametrów:

d = dict(p1=1, p2=2)
def f2(p1, p2):
    print(p1, p2)
f2(d)
Czy to możliwe?
Author: Martin Thoma, 2008-12-02

4 answers

W końcu sam to wymyśliłem. To proste, brakowało mi tylko operatora * * do rozpakowania słownika

Więc mój przykład staje się:

d = dict(p1=1, p2=2)
def f2(p1,p2):
    print p1, p2
f2(**d)
 569
Author: Dave Hillier,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2009-08-11 15:13:15
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

Kilka dodatkowych szczegółów, które mogą być pomocne do poznania (pytania miałem po przeczytaniu tego i poszedłem i przetestowałem):

  1. funkcja może mieć parametry, które są nie zawarte w słowniku
  2. możesz nie nadpisać parametr, który jest już w słowniku
  3. słownik może nie mieć parametry, których nie ma w funkcji.

Przykłady:

Numer 1: Na funkcja może mieć parametry, które nie są zawarte w słowniku

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

numer 2: nie można nadpisać parametru, który jest już w słowniku

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

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

numer 3: słownik nie może mieć parametrów, których nie ma w funkcji.

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

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

Zgodnie z prośbą w komentarzach, rozwiązanie numer 3 jest filtrowanie słownika na podstawie argumentów słów kluczowych dostępnych w funkcja:

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

Inną opcją jest akceptowanie (i ignorowanie) dodatkowych kwargów w funkcji:

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

Zwróć uwagę, że możesz używać argumentów pozycyjnych i list lub krotek w efektywny sposób, tak jak kwargs, oto bardziej zaawansowany przykład zawierający zarówno args pozycyjne, jak i słowa kluczowe:

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}
 145
Author: David Parks,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2020-03-09 17:42:47

W Pythonie nazywa się to" rozpakowywaniem", a trochę na ten temat znajdziesz w tutorialu. Dokumentacja jest do bani, zgadzam się, zwłaszcza ze względu na to, jak fantastycznie przydatna jest.

 35
Author: llimllib,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2014-04-28 21:24:43

Here ya go-działa tylko każdy inny iterable:

d = {'param' : 'test'}

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

f(d)
 7
Author: Patrick Harrington,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2008-12-02 16:52:20