Is it fine to raise an exception in __init__
in python? I have this piece of code:
class VersionManager(object): def __init__(self, path): self._path = path if not os.path.exists(path): os.mkdir(path) myfunction(path)
The second line can potentially result in an exception. In that case the object will not be init'ed properly. Is there a better way to handle situations where code in __init__
might throw an exception?
EDIT Added a call to a function after os.mkdir
Added a check to see if directory exists
The __init__ method is the Python equivalent of the C++ constructor in an object-oriented approach. The __init__ function is called every time an object is created from a class. The __init__ method lets the class initialize the object's attributes and serves no other purpose. It is only used within classes.
The associated value is usually passed as arguments to the exception class's constructor. User code can raise built-in exceptions.
In Python, exceptions can be handled using a try statement. The critical operation which can raise an exception is placed inside the try clause. The code that handles the exceptions is written in the except clause. We can thus choose what operations to perform once we have caught the exception.
__init__ : "__init__" is a reseved method in python classes. It is known as a constructor in object oriented concepts. This method called when an object is created from the class and it allow the class to initialize the attributes of a class.
It is perfectly fine to raise an exception in __init__
. You would then wrap the object initiation/creation call with try/except
and react to the exception.
One potential odd result though is that __del__
is run anyway:
class Demo(object): def __init__(self, value): self.value=value if value==2: raise ValueError def __del__(self): print '__del__', self.value d=Demo(1) # successfully create an object here d=22 # new int object labeled 'd'; old 'd' goes out of scope # '__del__ 1' is printed once a new name is put on old 'd' # since the object is deleted with no references
Now try with the value 2
that we are testing for:
Demo(2) Traceback (most recent call last): File "Untitled 3.py", line 11, in <module> Demo(2) File "Untitled 3.py", line 5, in __init__ raise ValueError ValueError __del__ 2 # But note that `__del__` is still run.
The creation of the object with value 2
raises a ValueError
exception and show that __del__
is still run to clean up the object.
Keep in mind that if you raise an exception during __init__
your object will not get a name. (It will, however, be created and destroyed. Since __del__
is paired with __new__
it still gets called)
ie, just like this does not create x
:
>>> x=1/0 Traceback (most recent call last): File "<stdin>", line 1, in <module> ZeroDivisionError: integer division or modulo by zero >>> x Traceback (most recent call last): File "<stdin>", line 1, in <module> NameError: name 'x' is not defined
Potential sneakier:
>>> x='Old X' >>> x=1/0 Traceback (most recent call last): File "<stdin>", line 1, in <module> ZeroDivisionError: division by zero >>> x 'Old X'
Same thing if you catch an exception of __init__
:
try: o=Demo(2) except ValueError: print o # name error -- 'o' never gets bound to the object... # Worst still -- 'o' is its OLD value!
So don't try to refer to the incomplete object o
-- it's gone out of scope by the time you get to except
. And the name o
is either nothing (i.e., NameError
if you try to use it) or its old value.
So wrapping up (thanks to Steve Jessop for the User Defined Exception idea), you can wrap the creation of the object and catch the exception. Just figure out how to react appropriately to the OS error you are looking at.
So:
class ForbiddenTwoException(Exception): pass class Demo(object): def __init__(self, value): self.value=value print 'trying to create with val:', value if value==2: raise ForbiddenTwoException def __del__(self): print '__del__', self.value try: o=Demo(2) except ForbiddenTwoException: print 'Doh! Cant create Demo with a "2"! Forbidden!!!' # with your example - react to being unusable to create a directory...
Prints:
trying to create with val: 2 Doh! Cant create Demo with a "2"! Forbidden!!! __del__ 2
You can wrap the call, as jramirez suggested:
try: ver = VersionManager(path) except: raise
Or you can use a context manager:
class VersionManager(object): def __init__(self): #not-so-harmful code self.path = path def __enter__(self): try: self.path = path os.mkdir(path) self.myfunction(path) except Exception as e: print e print "The directory making has failed, the function hasn't been executed." return self def __exit__(self, exc_type, exc_value, traceback): print(exc_type, exc_value, traceback)
And to run it:
with VersionManager(my_path) as myVersionManager: #do things you want with myVersionManager
This way, you'll catch errors inside the with
statement as well.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With