2011-04-01 8 views
26

En muchos idiomas podemos hacer algo como:¿Cuál es la mejor manera de omitir los valores N de la variable de iteración en Python?

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

¿Cómo puedo hacer lo mismo en Python? La siguiente (por supuesto) no funciona:

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

que podía hacer algo como esto:

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

pero me pregunto si hay una manera más elegante (Pythonic?) de hacer esto en Python.

Respuesta

20

Use continue.

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

Si desea forzar su iterable para saltar hacia adelante, debe llamar .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] 

Como puede ver, esto es repugnante.

+1

favor , revisa mi última edición Lo cambié para omitir 10 valores de * i * en lugar de solo uno. –

+0

Depende de lo que trates de hacer, de qué vuelcas, y por qué estás usando i como contador? La manera más pitonica puede ser no usar el "contador" en absoluto. Es difícil de decir sin un ejemplo más detallado. – monkut

+1

Los iteradores no le permiten 'saltar adelante' en una cantidad arbitraria de pasos. La única forma de saltar hacia adelante es llamar a '.next()' en su iterador la cantidad de veces que desea omitir. –

0

Yo creo que hay que utilizar un bucle while para esto ... para los bucles de bucle a través de una iterable..and no se puede saltar siguiente elemento como la forma en que desea hacerlo aquí

13

Crear el iterable antes del bucle .

Saltar uno utilizando el siguiente en el iterador

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

Saltar a muchos por el uso de itertools o recetas basadas en las ideas de itertools.

itertools.dropwhile()

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

Tome una lectura a través de la página itertools, que muestra algunos usos muy comunes de trabajo con iteradores.

itertools Islice

it = islice(xrange(value), 10) 
for i in it: 
    ...do stuff with i... 
+0

el primer código devuelve la siguiente excepción: 'TypeError: el objeto xrange no es un iterador'. Creo que será necesario especificar: 'it = xrange (value) .__ iter __()' –

+0

@oscar, thx. Solo estaba saliendo de mi cabeza. Simplemente envuélvelo en una llamada iter() para crear una instancia de un iterador. – kevpie

+1

@oscar, mientras que no fue exactamente lo que pudiste haber estado buscando, lo cambié a dropwhile. Si simplemente desea omitir un grupo de un iterador, islice puede hacerlo de manera muy afectiva si se trata de una incertidumbre. – kevpie

0

funciona un generador de aquí es rebundant? De esta manera:

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 
1

que espero que no estoy respondiendo a este mal ...pero esta es la forma más sencilla que he encontrado:

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

de salida debería ser algo como esto:

0 
2 
4 
6 
8 

La 2 en el parámetro gama de es el valor de salto

1

Hay unos pocos maneras a create iterators, pero el custom iterator class es el más extensible:

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 

y en uso:

>>> 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

itertools tiene una forma recomendada de hacer esto: 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)) 

Ahora usted puede hacer:

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

para obtener

5 
6 
7 
8 
9 
Cuestiones relacionadas