I have an object with an internal database connection that's active throughout its lifetime. At the end of the program's run, the connection has to be committed and closed. So far I've used an explicit close
method, but this is somewhat cumbersome, especially when exceptions can happen in the calling code.
I'm considering using the __del__
method for closing, but after some reading online I have concerns. Is this a valid usage pattern? Can I be sure that the internal resources will be freed in __del__
correctly?
This discussion raised a similar question but found no satisfactory answer. I don't want to have an explicit close
method, and using with
isn't an option, because my object isn't used as simply as open-play-close, but is kept as a member of another, larger object, that uses it while running in a GUI.
C++ has perfectly working destructors where one can free resources safely, so I would imagine Python has something agreed-upon too. For some reason it seems not to be the case, and many in the community vow against __del__
. What's the alternative, then?
You can make a connection module, since modules keep the same object in the whole application, and register a function to close it with the
atexit
moduleThis sugestion is based on the assumption that the connection in your application seems like a single instance (singleton) which a module global provides well.
If that's not the case, then you can use a destructor.
However destructors don't go well with garbage collectors and circular references (you must remove the circular reference yourself before the destructor is called) and if that's not the case (you need multiple connections) then you can go for a destructor. Just don't keep circular references around or you'll have to break them yourself.
Also, what you said about C++ is wrong. If you use destructors in C++ they are called either when the block that defines the object finishes (like python's
with
) or when you use thedelete
keyword (that deallocates an object created withnew
). Outside that you must use an explicitclose()
that is not the destructor. So it is just like python - python is even "better" because it has a garbage collector.Read up on the with statement. You're describing its use case.
You'll need to wrap your connection in a "Context Manager" class that handles the
__enter__
and__exit__
methods used by thewith
statement.See PEP 343 for more information.
Edit
"my object isn't used as simply as open-play-close, but is kept as a member of another, larger object"
Some folks call this "Dependency Injection" and "Inversion of Control". Other folks call this the Strategy pattern. The "ObjectThatMustBeClosed" is a strategy, plugged into some larger object. The assembly is created at a top-level of the GUI app, since that's usually where resources like databases are acquired.