Reputation: 7625
Pardon me if this question is too silly. The most common example of usefulness of using RAII is :
void func(){
// create some object pointer using any smart pointer
// do some operation that may throw
return;
}
// whether method returns from the *return* statement or because of any exception it is guaranteed that the memory will be released
This article says that (if I understood correctly), if runtime system knows that there is no exception handler that can catch an exception after being thrown it may skip calling the destructors of automatic objects.
There is also a proposed solution to that problem, namely use catch(..)
in main
.
Now my concern is if the proposed solution is not used, then there may be resource leak even after using RAII. And there are situation when the solution can not be applied ( like creating a library which will be used by others). In that case serious problem can occur like corrupting a file that contains valuable information.
Should we really be concerned about the problem? Or I am just missing something?
Upvotes: 6
Views: 745
Reputation: 67743
In order for your concern to be valid, you need some kind of resource which could be cleaned up by RAII, but which the OS won't clean up when std::terminate
is called and your process dies.
So, let's examine the sort of resources you could reasonably use RAII to clean up:
So the issue isn't generally with resources, which will generally be released by the OS, but semantics, where your RAII dtor was supposed to guarantee some clean state of a shared resource (shared memory, or files, or a network stream).
Should we really be concerned about the problem?
Well, we should be concerned about correct program semantics anyway. If your program has some external side-effects you need to guarantee, then a catch-all (at least around the relevant code) to guarantee RAII cleanup is the simplest of your concerns.
Upvotes: 6
Reputation: 153929
If you don't catch the exception, the program will terminate.
In this case, most of the resources you're worried about
(memory, mutex locks, etc.) will be cleaned up by the OS, so you
don't have to worry. The big exception is temporary files;
output files might also be an issue, since they may be
incomplete or inconsistent, and you don't want to leave
incomplete or inconsistent files lying around for someone to
accidentally use. (I usually use an OutputFile
class which
wraps an std::ofstream
, and deletes the file in the destructor
if commit
hasn't been called on it, or if the close
in
commit
fails.)
Of course, if there's an exception you don't expect, that's a serious error in the program; I've often found it useful for temporary files not to be deleted when I'm debugging, or trying to figure out why the code isn't working. (Such exceptions will never occur at a user site, of course, since you'll have sufficient tested the program before releasing it:-).)
If it is really an issue, you can use std::set_terminate
to
set a terminate handler, which can do any last minute clean-up.
(Be aware, however, that in this case, it is unspecified whether
the stack wil have been unwound or not, so you may have problems
determining what needs cleaning up.)
Upvotes: 2
Reputation: 145279
Re
“Should we really be concerned about the problem?”
it depends on the context.
If you’re using destructor breakpoints or logging in debugging, then you need the relevant destructor(s) to be called. Likewise if a destructor is saving crucial state for a "Bird of Phoenix" process re-instantiation after crash. And if possible it’s nice to have unneeded-for-recovery temporary files removed, not laying around after a crash.
On the other hand, since the solution is so utterly simple – a try
-catch
around some calling code, e.g. up in main
– it’s not really a practical problem. It’s more of just a thing to be aware of. E.g., to not expect destructors to necessarily be executed in someone else’s code that’s crashing via an unhandled exception.
Upvotes: 1
Reputation: 2149
I'm assuming here that you are not so much concerned about things such a memory leaks, but more about data corruption.
You'll need to analyze your design and application carefully, but I suppose it is theoretically possible that you might want a "last-chance" type of failsafe that would kick-in in the case of a serious program bug that would cause an uncaught exception. In which case, you could replace std::terminate
.
That said, if you are truly concerned about file corruption, what you describe would be inadequate. The way to prevent file corruption is by careful ordering of operations such as read/writes and proper flushing of file buffers (using fsync
or fdatasync
on Linux, for example) before considering an operation to be complete (i.e., committing an operation).
Upvotes: 2