Итерация / генератор использует значения из него (бесконечные генераторы являются исключением), что означает, что они больше не будут доступны на будущих итерациях (как вы видели). Для типичного итератора / генератора в Python единственный верный способ «перезапустить» - это повторно инициализировать его.
>>> sol = map(pow, [1, 2, 3], [4, 5, 6])
>>> list(sol)
[1, 32, 729]
>>> next(sol)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
StopIteration
>>> sol = map(pow, [1, 2, 3], [4, 5, 6])
>>> next(sol)
1
Существуют способы работы с итератором, чтобы сделать его многоразовым хотя, например, с itertools.tee
(как упомянуто в одном из ответов на вопрос, связанный с @JanChristophTerasa), или для преобразования итератора в список, который сохранит свои данные.
itertools.tee
>>> from itertools import tee
>>> sol = map(pow, [1, 2, 3], [4, 5, 6])
>>> a, b = tee(sol, 2)
>>> list(a)
[1, 32, 729]
>>> list(b)
[1, 32, 729]
>>> list(a)
[]
с tee
, хотя и a
, и b
все равно будут итераторами, поэтому у вас возникнет та же проблема с ними.
Другой распространенный способ справиться с этим - list()
sol = list(map(pow, [1, 2, 3], [4, 5, 6]))
>>> sol
[1, 32, 729]
>>> sol
[1, 32, 729]
Теперь sol
- это список значений вместо итератора, что означает, что вы можете повторять его столько раз, сколько хотите - значения останется там. Это означает , что означает, что вы не можете использовать next
с ним (в смысле next(sol)
), но вы можете получить итератор из вашего нового списка с помощью iter(sol)
, если вам нужен итератор специально .
Редактировать
Я видел itertools.cycle
, упомянутый в комментариях, что также является допустимым вариантом, поэтому я подумал, что мог бы также добавить сюда некоторую информацию.
itertools.cycle
- один из тех бесконечных генераторов, которые я упоминал в начале. Это все еще итератор , но таким образом, что у вас никогда не кончатся значения.
>>> from itertools import cycle
>>> sol = map(pow, [1, 2, 3], [4, 5, 6])
>>> infinite = cycle(sol)
>>> for _ in range(5):
... print(next(infinite))
...
1
32
729
1
32
>>>
Несколько замечаний по этому поводу - после итерации infinite
N раз будет располагаться после того, как из него было извлечено последнее значение. Повторная итерация позже возобновит с этой позиции, не с начала.
Кроме того, и это очень важно, не повторяется бесконечный генератор неограниченным образом, например, list(infinite)
или for x in infinite:
, иначе у вас будет плохое время.