函数调用中的星号

Ram*_*amy 105 python operators

我正在使用itertools.chain以这种方式"压扁"列表列表:

uniqueCrossTabs = list(itertools.chain(*uniqueCrossTabs))
Run Code Online (Sandbox Code Playgroud)

这有什么不同于说:

uniqueCrossTabs = list(itertools.chain(uniqueCrossTabs))
Run Code Online (Sandbox Code Playgroud)

Cam*_*ron 174

* 是"splat"运算符:它将列表作为输入,并将其扩展为函数调用中的实际位置参数.

如果uniqueCrossTabs是的话[ [ 1, 2 ], [ 3, 4 ] ],那itertools.chain(*uniqueCrossTabs)就像说的一样itertools.chain([ 1, 2 ], [ 3, 4 ])

这明显不同于传入uniqueCrossTabs.在您的情况下,您有一个您希望展平的列表列表; 什么itertools.chain()是在你传递给它的所有位置参数的串联上返回一个迭代器,其中每个位置参数本身是可迭代的.

换句话说,您希望将每个列表uniqueCrossTabs作为参数传递给chain()它们,它们将它们链接在一起,但是您没有单独变量中的列表,因此您使用*运算符将列表列表扩展为多个列表参数.

正如Jochen Ritzel在评论中指出的那样,chain.from_iterable()它更适合于这种操作,因为它假定可以迭代一次迭代.您的代码变得简单:

uniqueCrossTabs = list(itertools.chain.from_iterable(uniqueCrossTabs))
Run Code Online (Sandbox Code Playgroud)

  • +1,我不知道他们叫它*splat*. (46认同)
  • @larsmans:我认为这个术语在Ruby世界中更受欢迎,但它似乎也是[Python也可以接受](http://stackoverflow.com/questions/2322355/proper-name-for-python-operator)I喜欢它,因为说它很有趣;-) (9认同)

Ign*_*ams 68

它将序列拆分为函数调用的单独参数.

>>> def foo(a, b=None, c=None):
...   print a, b, c
... 
>>> foo([1, 2, 3])
[1, 2, 3] None None
>>> foo(*[1, 2, 3])
1 2 3
>>> def bar(*a):
...   print a
... 
>>> bar([1, 2, 3])
([1, 2, 3],)
>>> bar(*[1, 2, 3])
(1, 2, 3)
Run Code Online (Sandbox Code Playgroud)


Gel*_*der 26

只是解释概念/使用它的另一种方式.

import random

def arbitrary():
    return [x for x in range(1, random.randint(3,10))]

a, b, *rest = arbitrary()

# a = 1
# b = 2
# rest = [3,4,5]
Run Code Online (Sandbox Code Playgroud)

  • 这很重要,其他地方没有提到 (3认同)
  • 这个答案并不特别适用于这个问题,但它是星号的一个重要应用(所以我认为在相当“模糊”的标题下是合适的)。同样,另一个重要的应用是函数定义:`def func(a, b, *args):` 有关更多信息,请参阅[此答案](http://stackoverflow.com/a/9539977/6748105)。 (2认同)