Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to look ahead one element (peek) in a Python generator?

I can't figure out how to look ahead one element in a Python generator. As soon as I look it's gone.

Here is what I mean:

gen = iter([1,2,3]) next_value = gen.next()  # okay, I looked forward and see that next_value = 1 # but now: list(gen)  # is [2, 3]  -- the first value is gone! 

Here is a more real example:

gen = element_generator() if gen.next_value() == 'STOP':   quit_application() else:   process(gen.next()) 

Can anyone help me write a generator that you can look one element forward?

like image 299
bodacydo Avatar asked Mar 11 '10 13:03

bodacydo


2 Answers

For sake of completeness, the more-itertools package (which should probably be part of any Python programmer's toolbox) includes a peekable wrapper that implements this behavior. As the code example in the documentation shows:

>>> p = peekable(['a', 'b']) >>> p.peek() 'a' >>> next(p) 'a' 

However, it's often possible to rewrite code that would use this functionality so that it doesn't actually need it. For example, your realistic code sample from the question could be written like this:

gen = element_generator() command = gen.next_value() if command == 'STOP':   quit_application() else:   process(command) 

(reader's note: I've preserved the syntax in the example from the question as of when I'm writing this, even though it refers to an outdated version of Python)

like image 177
David Z Avatar answered Sep 23 '22 15:09

David Z


The Python generator API is one way: You can't push back elements you've read. But you can create a new iterator using the itertools module and prepend the element:

import itertools  gen = iter([1,2,3]) peek = gen.next() print list(itertools.chain([peek], gen)) 
like image 35
Aaron Digulla Avatar answered Sep 24 '22 15:09

Aaron Digulla