28

Suppose we have an iterator (an infinite one) that returns lists (or finite iterators), for example one returned by

infinite = itertools.cycle([[1,2,3]])

Note: it is not the same problem as just flattening a nested list - solutions which try to exhaustively process the list will not work, e.g. [item for sublist in infinite for item in sublist] will exhaust memory.

What is a good Python idiom to get an iterator (obviously infinite) that will return each of the elements from the first iterator, then each from the second one, etc. In the example above it would return 1,2,3,1,2,3,.... The iterator is infinite, so itertools.chain(*infinite) will not work.

Related

Rafał Dowgird
  • 43,216
  • 11
  • 77
  • 90

2 Answers2

55

Starting with Python 2.6, you can use itertools.chain.from_iterable:

itertools.chain.from_iterable(iterables)

You can also do this with a nested generator comprehension:

def flatten(iterables):
    return (elem for iterable in iterables for elem in iterable)
Mark Amery
  • 143,130
  • 81
  • 406
  • 459
Torsten Marek
  • 83,780
  • 21
  • 91
  • 98
13

Use a generator:

(item for it in infinite for item in it)

The * construct unpacks into a tuple in order to pass the arguments, so there's no way to use it.

Thomas Wouters
  • 130,178
  • 23
  • 148
  • 122