[Python-ideas] Yield-From: Finalization guarantees

Jacob Holm jh at improva.dk
Wed Apr 1 17:39:34 CEST 2009


Nick Coghlan wrote:
> Jacob Holm wrote:
>   
>> Explicitly catching GeneratorExit and then returning is a valid use
>> today that I wouldn't consider suspect.  Catching GeneratorExit and then
>> exiting the except block by other means than a raise or return is
>> suspect, but has valid uses.
>>     
>
> What are these valid uses? The PEP 342 definition made some sense
> originally when GeneratorExit was a subclass of Exception and hence easy
> to suppress accidentally, but I have serious doubts about the validity
> of trapping it and turning it into StopIteration now that it has been
> moved out to inherit directly from BaseException.
>   
When catching and returning, the control flow is different than if you 
were catching and raising.  Also using "return" more clearly signals the 
intent to leave the generator than "raise".  Even with GeneratorExit 
inheriting directly from BaseException, it is still easier to intercept 
an exception than a return.

The use for catching and exiting the block normally is to share some 
code between the cases.  Of course you need to be careful when you do 
that, but it can save some duplication.

Both these uses are valid in the sense that the generators work as 
advertised and follow the rules of finalization as defined by PEP 342.  
I don't think a proposal for changing close to not accept StopIteration 
is going to fly.

> Regardless, unless Greg goes out of his way to change the meaning of
> close() in the PEP, GeneratorReturn will escape from close() (since that
> only traps StopIteration). That means you'll be able to catch that
> exception directly if you really want to, and if you don't it will
> bubble up out of the original close() call that was made on the
> outermost generator.
>   

Hmm.  I had almost forgotten about the separate GeneratorReturn 
exception.  It would be good to see how that changes  things.  So far I 
consider it a needless complication, but I would like to read a version 
of the PEP that include it to see how bad it is.

As for GeneratorReturn not being caught by close(), I find it really 
strange if returning a non-None value as a response to GeneratorExit 
makes close() raise a GeneratorReturn.  Whereas returning None makes 
close finish without an exception.  If you think returning a non-None 
value is an error, we should make it a (subclass of) RuntimeError rather 
than a GeneratorReturn to clearly indicate this.

I am strongly in favor of changing close to return the value rather than 
letting the GeneratorReturn pass through or raising a RuntimeError.  I 
think the "averager" example I just gave is a good (but simplistic) 
example of the kind of code I would consider using coroutines for.  The 
need to catch an exception would make that code a lot less readable, not 
to mention slower. 

I am not about to write a separate PEP for this, but I would consider 
"return from generator" plus "close returns value returned from 
generator"  to be a worthwhile addition in itself.

- Jacob





More information about the Python-ideas mailing list