Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to repeat each of a Python list's elements n times with itertools only?

I have a list with numbers: numbers = [1, 2, 3, 4].

I would like to have a list where they repeat n times like so (for n = 3):

[1, 1, 1, 2, 2, 2, 3, 3, 3, 4, 4, 4].

The problem is that I would like to only use itertools for this, since I am very constrained in performance.

I tried to use this expression:

list(itertools.chain.from_iterable(itertools.repeat(numbers, 3)))

But it gives me this kind of result:

[1, 2, 3, 4, 1, 2, 3, 4, 1, 2, 3, 4]

which is obviously not what I need.

Is there a way to do this with itertools only, without using sorting, loops and list comprehensions? The closest I could get is:

list(itertools.chain.from_iterable([itertools.repeat(i, 3) for i in numbers])),

but it also uses list comprehension, which I would like to avoid.

like image 779
Sergey Zakharov Avatar asked Aug 21 '17 14:08

Sergey Zakharov


2 Answers

Firstly, using functions from itertools won't necessarily be faster than a list comprehension: you should benchmark.

Pure list comprehension approach:

>>> numbers = [1, 2, 3, 4]
>>> [y for x in numbers for y in (x,)*3]
[1, 1, 1, 2, 2, 2, 3, 3, 3, 4, 4, 4]

Using chain.from_iterable() with repeat() in a generator expression:

>>> from itertools import chain, repeat
>>> list(chain.from_iterable(repeat(n, 3) for n in numbers))
[1, 1, 1, 2, 2, 2, 3, 3, 3, 4, 4, 4]

Using chain.from_iterable() and zip():

>>> from itertools import chain
>>> list(chain.from_iterable(zip(*(numbers,)*3)))
[1, 1, 1, 2, 2, 2, 3, 3, 3, 4, 4, 4]
like image 107
Eugene Yarmash Avatar answered Oct 11 '22 14:10

Eugene Yarmash


Since you don't want to use list comprehension, following is a pure (+zip) itertools method to do it -

from itertools import chain, repeat

list(chain.from_iterable(zip(*repeat(numbers, 3))))
# [1, 1, 1, 2, 2, 2, 3, 3, 3, 4, 4, 4]
like image 22
shad0w_wa1k3r Avatar answered Oct 11 '22 13:10

shad0w_wa1k3r