Каков наилучший способ пропустить N значений переменной итерации в Python? - PullRequest
27 голосов
/ 01 апреля 2011

Во многих языках мы можем сделать что-то вроде:

for (int i = 0; i < value; i++)
{
    if (condition)
    {
        i += 10;
    }
}

Как я могу сделать то же самое в Python? Следующее (конечно) не работает:

for i in xrange(value):
    if condition:
        i += 10

Я мог бы сделать что-то вроде этого:

i = 0
while i < value:
  if condition:
    i += 10
  i += 1

но мне интересно, есть ли более элегантный ( pythonic? ) способ сделать это в Python.

Ответы [ 7 ]

24 голосов
/ 01 апреля 2011

Использование continue.

for i in xrange(value):
    if condition:
        continue

Если вы хотите, чтобы ваша итерация пропускалась вперед, вы должны позвонить .next().

>>> iterable = iter(xrange(100))
>>> for i in iterable:
...     if i % 10 == 0:
...         [iterable.next() for x in range(10)]
... 
[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
[21, 22, 23, 24, 25, 26, 27, 28, 29, 30]
[41, 42, 43, 44, 45, 46, 47, 48, 49, 50]
[61, 62, 63, 64, 65, 66, 67, 68, 69, 70]
[81, 82, 83, 84, 85, 86, 87, 88, 89, 90]

отвратительно.

16 голосов
/ 01 апреля 2011

Создайте итерацию перед циклом.

Пропустите одну, используя следующую на итераторе

it = iter(xrange(value))
for i in it:
    if condition:
        i = next(it)

Пропустите многие, используя itertools или рецепты, основанные на идеях из itertools.

itertools.drop while ()

it = iter(xrange(value))
for i in it:
    if x<5:
        i = dropwhile(lambda x: x<5, it)

Прочитайте страницу itertools, она показывает очень распространенное использование работы с итераторами.

itertools islice

it = islice(xrange(value), 10)
for i in it:
    ...do stuff with i...
2 голосов
/ 05 ноября 2015

Я надеюсь, что не отвечу на этот вопрос неправильно ... но это самый простой способ, с которым я столкнулся:

for x in range(0,10,2):
    print x

вывод должен выглядеть примерно так:

0
2
4
6
8

2 в параметре диапазона - это значение скачка

1 голос
/ 27 января 2017

Itertools имеет рекомендуемый способ сделать это: https://docs.python.org/3.7/library/itertools.html#itertools-recipes

import collections
def tail(n, iterable):
    "Return an iterator over the last n items"
    # tail(3, 'ABCDEFG') --> E F G
    return iter(collections.deque(iterable, maxlen=n))

Теперь вы можете сделать:

for i in tail(5, range(10)):
    print(i)

, чтобы получить

5
6
7
8
9
1 голос
/ 26 марта 2016

Существует несколько способов создания итераторов , но пользовательский класс является наиболее расширяемым:

class skip_if:   # skip_if(object) for python2
    """
    iterates through iterable, calling skipper with each value
    if skipper returns a positive integer, that many values are
    skipped
    """
    def __init__(self, iterable, skipper):
        self.it = iter(iterable)
        self.skip = skipper
    def __iter__(self):
        return self
    def __next__(self):   # def next(self): for python2
        value = next(self.it)
        for _ in range(self.skip(value)):
            next(self.it, None)
        return value

и используется:

>>> for i in skip_if(range(1,100), lambda n: 10 if not n%10 else 0):
...   print(i, end=', ')
... 
 1,  2,  3,  4,  5,  6,  7,  8,  9, 10,
21, 22, 23, 24, 25, 26, 27, 28, 29, 30,
41, 42, 43, 44, 45, 46, 47, 48, 49, 50,
61, 62, 63, 64, 65, 66, 67, 68, 69, 70,
81, 82, 83, 84, 85, 86, 87, 88, 89, 90,
0 голосов
/ 01 апреля 2011

Является ли функция генератора здесь избыточной? Как это:

def filterRange(range, condition):
x = 0
while x < range:
    x = (x+10) if condition(x) else (x + 1)
    yield x

if __name__ == "__main__":
for i in filterRange(100, lambda x: x > 2):
    print i
0 голосов
/ 01 апреля 2011

Я думаю, вы должны использовать цикл while для этого ... для циклов цикла над итерируемым ... и вы не можете пропустить следующий элемент, например, как вы хотите это сделать здесь

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...