Answers:
*
是“ splat”运算符:它接受一个列表作为输入,并将其扩展为函数调用中的实际位置参数。
所以如果uniqueCrossTabs
是[ [ 1, 2 ], [ 3, 4 ] ]
,那就itertools.chain(*uniqueCrossTabs)
等于说itertools.chain([ 1, 2 ], [ 3, 4 ])
这与传递just显然不同uniqueCrossTabs
。对于您的情况,您有一个想要拼合的列表列表;什么itertools.chain()
确实是在所有你传递给它的位置参数,其中每个位置参数是在自己的权利迭代拼接返回一个迭代。
换句话说,您希望将每个列表uniqueCrossTabs
作为参数传递给chain()
,这会将它们链接在一起,但是您没有在单独的变量中使用列表,因此可以使用*
运算符将列表扩展为多个列表参数。
正如Jochen Ritzel在评论中指出的那样,chain.from_iterable()
它更适合于此操作,因为它假定一个可迭代的对象开始。然后,您的代码将变得简单:
uniqueCrossTabs = list(itertools.chain.from_iterable(uniqueCrossTabs))
chain(*it)
我写chain.from_iterable(it)
。
from_iterable
存在!我将添加到我的答案不久
*
仅用于将列表分解为函数的位置参数(因此,非常具体)。您可以for l in uniqueCrossTabs:
对它们进行迭代。不幸的是,很难看到*
它在工作,因为它仅在将列表传递给函数时才起作用(而不是将列表作为第一个参数传递,而是*
导致列表中的每个元素作为一个单独的参数传递,一个接一个地传递) ,就好像在参数列表中用逗号隔开了它们一样)
它将序列拆分为函数调用的单独参数。
>>> 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)
只是解释概念/使用它的另一种方法。
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]