I have been bitten by something unexpected recently. I wanted to make something like that:
try : thing.merge(iterable) # this is an iterable so I add it to the list except TypeError : thing.append(iterable) # this is not iterable, so I add it
Well, It was working fine until I passed an object inheriting from Exception which was supposed to be added.
Unfortunetly, an Exception is iterable. The following code does not raise any TypeError:
for x in Exception() : print 1
Does anybody know why?
Note that what is happening is not related to any kind of implicit string conversion etc, but because the
Exceptionclass implements__getitem__to return the values from the args tuple (ex.args). You can see this by the fact that you get the whole string as your first and only item in the iteration, rather than the character-by-character result you’d get if you iterate over the string.This surprised me too, but thinking about it, I’m guessing it is for backwards compatibility reasons. Python used to (pre-1.5) lack the current class hierarchy of exceptions. Instead, strings were thrown, with (usually) a tuple argument for any details that should be passed to the handling block, i.e:
It looks like this behavior was put in to avoid breaking pre-1.5 code expecting a tuple of arguments, rather than a non-iterable exception object. There are a couple of examples of this with
IOErrorin the Fatal Breakage section of the above linkString exceptions have been deprecated for a while, and are gone in Python 3. Exception objects are no longer iterable in Python 3: