Что означают * tuple и ** dict в Python?

Как упоминалось в PythonCookbook, * можно добавить перед кортежем. Что здесь *?

Глава 1.18. Сопоставление имен с элементами последовательности:

from collections import namedtuple
Stock = namedtuple('Stock', ['name', 'shares', 'price'])
s = Stock(*rec) 
# here rec is an ordinary tuple, for example: rec = ('ACME', 100, 123.45)

В этом же разделе **dict представляет:

from collections import namedtuple
Stock = namedtuple('Stock', ['name', 'shares', 'price', 'date', 'time'])
# Create a prototype instance
stock_prototype = Stock('', 0, 0.0, None, None)
# Function to convert a dictionary to a Stock
def dict_to_stock(s):
    return stock_prototype._replace(**s)

Какая здесь функция **?


person heLomaN    schedule 16.02.2014    source источник
comment
Прочтите руководство по Python. (Распаковка списков аргументов)   -  person falsetru    schedule 16.02.2014
comment
И это не **tuple, а **dictionary.   -  person Martijn Pieters    schedule 16.02.2014
comment
@MartijnPieters ИЗВИНИТЕ за это, я постараюсь это исправить.   -  person heLomaN    schedule 16.02.2014
comment
Нет проблем, просто указываю на недоразумение с вашей стороны.   -  person Martijn Pieters    schedule 16.02.2014
comment
@MartijnPieters Это не повторяющийся вопрос, если вы начинаете с языка. Для меня этот вопрос дал другую точку зрения, чем * arg и ** kwargs.   -  person Anurag Jain    schedule 03.07.2018


Ответы (1)


В вызове функции

*t означает обрабатывать элементы этого итеративного объекта как позиционные аргументы для этого вызова функции.

def foo(x, y):
    print(x, y)

>>> t = (1, 2)
>>> foo(*t)
1 2

Начиная с v3.5, вы также можете сделать это в литералах list / tuple / set:

>>> [1, *(2, 3), 4]
[1, 2, 3, 4]

**d означает рассматривать пары ключ-значение в словаре как дополнительные именованные аргументы для этого вызова функции.

def foo(x, y):
    print(x, y)

>>> d = {'x':1, 'y':2}
>>> foo(**d)
1 2

Начиная с v3.5, вы также можете сделать это в словарных литералах:

>>> d = {'a': 1}
>>> {'b': 2, **d}
{'b': 2, 'a': 1}

В сигнатуре функции

*t означает взять все дополнительные позиционные аргументы этой функции и упаковать их в этот параметр как кортеж.

def foo(*t):
    print(t)

>>> foo(1, 2)
(1, 2)

**d означает взять все дополнительные именованные аргументы этой функции и вставить их в этот параметр как словарные статьи.

def foo(**d):
    print(d)

>>> foo(x=1, y=2)
{'y': 2, 'x': 1}

В заданиях и for петлях

*x означает потребление дополнительных элементов в правой части, но это не обязательно должен быть последний элемент. Обратите внимание, что x всегда будет списком:

>>> x, *xs = (1, 2, 3, 4)
>>> x
1
>>> xs
[2, 3, 4]

>>> *xs, x = (1, 2, 3, 4)
>>> xs
[1, 2, 3]
>>> x
4

>>> x, *xs, y = (1, 2, 3, 4)
>>> x
1
>>> xs
[2, 3]
>>> y
4

>>> for (x, *y, z) in [ (1, 2, 3, 4) ]: print(x, y, z)
...
1 [2, 3] 4

Обратите внимание, что параметры, которые появляются после *, относятся только к ключевым словам:

def f(a, *, b): ...

f(1, b=2)  # fine
f(1, 2)    # error: b is keyword-only

Python3.8 добавил только позиционные параметры, то есть параметры которые нельзя использовать в качестве аргументов ключевого слова. Они появляются перед / (каламбур на * предшествующих аргументах, содержащих только ключевые слова).

def f(a, /, p, *, k): ...

f(  1,   2, k=3)  # fine
f(  1, p=2, k=3)  # fine
f(a=1, p=2, k=3)  # error: a is positional-only
person Elazar    schedule 16.02.2014
comment
Хороший ответ. Я просто добавлю ключевое слово operator, чтобы облегчить поиск этого ответа, когда кто-то ищет python operator **, ожидая * или **, в данном контексте это называется оператором. - person Mikko Rantalainen; 05.01.2021
comment
Отличный ответ. Просто добавлю, что в случае сигнатуры функции общей идиомой является использование * args для позиционных аргументов и ** kwargs для аргументов ключевого слова. - person Guzman Ojero; 26.01.2021
comment
@GuzmanOjero, это правда, но только если нет значимой альтернативы. - person Elazar; 26.01.2021