在函数调用中,星号运算符是什么意思?


630

*运算符在Python中是什么意思,例如likezip(*x)或代码f(**k)

  1. 在解释器内部如何处理?
  2. 它会影响性能吗?是快还是慢?
  3. 什么时候有用,什么时候没有?
  4. 应该在函数声明中还是在调用中使用它?


4
我认为这应该表述为“ *函数调用语法”。他们不是运营商,但它会被迷惑,因为一个***运营商什么都没有做这句法。
伊恩·比金

1
@Ian Bicking:您完全正确,参数列表中的*和**是纯语法(令牌)。
P. Ortiz

2
注意:对于PEP 448:额外的Unpacking Generalizations特定内容(例如[*a, b, *c]{**d1, **d2}),您需要在元组,列表和集合定义中阅读星号,在dict定义中阅读双星号,这特定于函数调用和函数定义之外的用法。对于较早的PEP 3132如果您不知道序列长度,请参阅Python中的多重开箱分配
ShadowRanger

1
VTR-这不是**(双星号/星号)和*(星号/星号)对参数有何作用的重复项由于该问题仅与参数有关,即使答案也涵盖了函数调用。函数调用中的星号应被标记为该问题的重复项,因为它不那么受欢迎,且最不完整的答案也是如此。
wjandrea

Answers:


966

单颗星*将序列/集合解压缩为位置参数,因此您可以执行以下操作:

def sum(a, b):
    return a + b

values = (1, 2)

s = sum(*values)

这将打开元组的包装,使其实际执行为:

s = sum(1, 2)

双星**只使用字典并因此命名参数来做同样的事情:

values = { 'a': 1, 'b': 2 }
s = sum(**values)

您还可以结合:

def sum(a, b, c, d):
    return a + b + c + d

values1 = (1, 2)
values2 = { 'c': 10, 'd': 15 }
s = sum(*values1, **values2)

将执行为:

s = sum(1, 2, c=10, d=15)

另请参见Python文档的4.7.4-解包参数列表


另外,您可以定义要接受的函数*x**y参数,这使函数可以接受在声明中未专门命名的任何数量的位置和/或命名参数。

例:

def sum(*values):
    s = 0
    for v in values:
        s = s + v
    return s

s = sum(1, 2, 3, 4, 5)

或搭配**

def get_a(**values):
    return values['a']

s = get_a(a=1, b=2)      # returns 1

这可以使您无需声明它们即可指定大量可选参数。

再一次,您可以结合:

def sum(*values, **options):
    s = 0
    for i in values:
        s = s + i
    if "neg" in options:
        if options["neg"]:
            s = -s
    return s

s = sum(1, 2, 3, 4, 5)            # returns 15
s = sum(1, 2, 3, 4, 5, neg=True)  # returns -15
s = sum(1, 2, 3, 4, 5, neg=False) # returns 15

4
您为什么需要它,函数不能在不扩展的情况下仅遍历提供的列表?
马丁·贝克特

30
当然可以,但是您必须调用它:s = sum((1, 2, 3, 4, 5))s = sum([1, 2, 3, 4, 5]),该*values选项使调用看起来像需要多个参数,但它们被打包到函数代码的集合中。
拉瑟五世

12
这是真正的好处:如果需要可变数量的参数,则可以编写其他方式无法实现的功能。例如,C的具有1 + n个参数的printf函数对于任何初学者来说都是很难编写的。在Python中,初学者可以编写def printf(string_template,* args)并继续前进。
IceArdor

1
如果您(偶然地:p)只用一个*而不是两个来解开字典,会发生什么?它似乎在做某事,看起来像一个元组出来,但是它并不是那么明显。(编辑:确定,我想答案是,它只是解包了键,值被丢弃了)
Ben Farmer

1
最后一个示例表明*和**不仅在进行拆包,而且还在进行打包!看到这个优秀的页面codingame.com/playgrounds/500/...
HCChen

46

一点:这些不是运算符。表达式中使用运算符从现有值创建新值(例如1 + 2变为3。这里的*和**是函数声明和调用语法的一部分。


7
请注意,在这种情况下,Python文档确实将*称为运算符;我同意,这是一种误导。
Christophe

谢谢。我一直在python参考文档中寻求对此的明确说明,但仍然没有看到它。因此,函数调用的规则基本上是,在函数调用内表达式的开头处的“ *”或“ **”会导致这种扩展吗?
nealmcb 2012年

21

对于要“存储”函数调用的情况,我发现这特别有用。

例如,假设我对功能“ add”进行了一些单元测试:

def add(a, b): return a + b
tests = { (1,4):5, (0, 0):0, (-1, 3):3 }
for test, result in tests.items():
   print 'test: adding', test, '==', result, '---', add(*test) == result

除了手动执行类似add(test [0],test [1])之类的丑陋操作外,没有其他方法可以调用add。另外,如果变量数量可变,则所有需要的if语句的代码都会变得很丑陋。

另一个有用的地方是定义Factory对象(为您创建对象的对象)。假设您有一些工厂类,该类使Car对象返回。您可以使myFactory.make_car('red','bmw','335ix')创建Car('red','bmw','335ix'),然后返回它。

def make_car(*args):
   return Car(*args)

当您要调用超类的构造函数时,这也很有用。


4
我喜欢你的例子。但是,我认为1 + 3 == 2
eksortso

5
我特意在其中放置了可能会失败的内容:)
唐纳德·迈纳

19

它称为扩展调用语法。从文档中

如果语法* expression出现在函数调用中,则表达式必须计算为序列。来自此序列的元素被视为它们是附加的位置参数。如果存在位置参数x1,...,xN,并且表达式的计算结果为序列y1,...,yM,则等效于使用M + N个位置参数x1,...,xN,y1,...的调用。 ..,yM。

和:

如果语法** expression出现在函数调用中,则expression必须计算为一个映射,该映射的内容被视为其他关键字参数。如果关键字同时出现在表达式中并作为显式关键字参数出现,则会引发TypeError异常。


3
只需在教科书答案中添加脚注-在语法支持到来之前,内置功能就实现了相同的apply()功能
杰里米·布朗

18

在函数调用中,单星号将列表变成单独的参数(例如zip(*x),与zip(x1,x2,x3)if相同x=[x1,x2,x3]),而双星号将字典变成单独的关键字参数(例如f(**k),与f(x=my_x, y=my_y)if相同)k = {'x':my_x, 'y':my_y}

在函数定义中,反之亦然:单星将任意数量的参数转换为列表,而双引号将任意数量的关键字参数转换为字典。例如,def foo(*x)表示“ foo接受任意数量的参数,可以通过列表x来访问它们(即,如果用户调用foo(1,2,3)x将是[1,2,3])”,并且def bar(**k)表示“ bar可以接受任意数量的关键字参数,并且可以通过字典k进行访问。 (即,如果用户致电bar(x=42, y=23)k将为{'x': 42, 'y': 23})”。


3
一个(非常)晚的评论,但我相信def foo(*x)* x给出的是一个元组,而不是列表。
jeremycg '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.