*
is the "splat" operator: It takes a list as input, and expands it into actual positional arguments in the function call.
So if uniqueCrossTabs
was [ [ 1, 2 ], [ 3, 4 ] ]
, then itertools.chain(*uniqueCrossTabs)
is the same as saying itertools.chain([ 1, 2 ], [ 3, 4 ])
This is obviously different from passing in just uniqueCrossTabs
. In your case, you have a list of lists that you wish to flatten; what itertools.chain()
does is return an iterator over the concatenation of all the positional arguments you pass to it, where each positional argument is iterable in its own right.
In other words, you want to pass each list in uniqueCrossTabs
as an argument to chain()
, which will chain them together, but you don't have the lists in separate variables, so you use the *
operator to expand the list of lists into several list arguments.
As Jochen Ritzel has pointed out in the comments, chain.from_iterable()
is better-suited for this operation, as it assumes a single iterable of iterables to begin with. Your code then becomes simply:
uniqueCrossTabs = list(itertools.chain.from_iterable(uniqueCrossTabs))
It splits the sequence into separate arguments for the function call.
>>> 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)
Just an alternative way of explaining the concept/using it.
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]
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With