Функция разностного вызова с параметром asterisk и без
Я знаю, что значение звездочки находится в определении функции в Python.
Я часто, однако, вижу звездочки для вызовов функций с такими параметрами, как:
def foo(*args, **kwargs):
first_func(args, kwargs)
second_func(*args, **kwargs)
В чем разница между первым и вторым вызовами функции?
Ответы
Ответ 1
Пусть args = [1,2,3]
:
func(*args) == func(1,2,3)
- переменные распаковываются из списка (или любого другого типа последовательности) в качестве параметров
func(args) == func([1,2,3])
- список передан
Пусть kwargs = dict(a=1,b=2,c=3)
:
func(kwargs) == func({'a':1, 'b':2, 'c':3})
- диктат пройден
func(*kwargs) == func(('a','b','c'))
- кортеж из клавиш dict (в случайном порядке)
func(**kwargs) == func(a=1,b=2,c=3)
- (ключ, значение) распаковываются из dict (или любого другого типа отображения) как именованные параметры
Ответ 2
Различие заключается в том, как аргументы передаются в вызываемые функции. Когда вы используете *
, аргументы распаковываются (если они представляют собой список или кортеж); в противном случае они просто передаются как есть.
Вот пример разницы:
>>> def add(a, b):
... print a + b
...
>>> add(*[2,3])
5
>>> add([2,3])
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: add() takes exactly 2 arguments (1 given)
>>> add(4, 5)
9
Когда я префикс аргумента *
, он фактически распаковал список на два отдельных аргумента, которые были переданы в add
как a
и b
. Без него он просто передается в списке как один аргумент.
То же самое относится к словарям и **
, за исключением того, что они передаются как имена, а не упорядоченные аргументы.
>>> def show_two_stars(first, second='second', third='third'):
... print "first: " + str(first)
... print "second: " + str(second)
... print "third: " + str(third)
>>> show_two_stars('a', 'b', 'c')
first: a
second: b
third: c
>>> show_two_stars(**{'second': 'hey', 'first': 'you'})
first: you
second: hey
third: third
>>> show_two_stars({'second': 'hey', 'first': 'you'})
first: {'second': 'hey', 'first': 'you'}
second: second
third: third
Ответ 3
def fun1(*args):
""" This function accepts a non keyworded variable length argument as a parameter.
"""
print args
print len(args)
>>> a = []
>>> fun1(a)
([],)
1
# This clearly shows that, the empty list itself is passed as a first argument. Since *args now contains one empty list as its first argument, so the length is 1
>>> fun1(*a)
()
0
# Here the empty list is unwrapped (elements are brought out as separate variable length arguments) and passed to the function. Since there is no element inside, the length of *args is 0
>>>