Reputation: 101
I have a simple C++ object that I create at the start of function F()
to ensure two matched functions (OpDo, OpUndo) are called at the start and return of the F()
, by using the object's constructor and destructor. However, I don't want the operation to be undone in case an exception was thrown within the body of F()
. Is this possible to do cleanly? I have read about std::uncaught_exception
, but its use does not seem to be recommended.
Upvotes: 7
Views: 1531
Reputation: 131799
You can subvert the Scope Guard idiom. Instead of not doing something in the destructor when no exception is thrown, we invert that and only do something if no exception is thrown:
class DoUndoRAII{
public:
DoUndoRAII()
: noexcept_(false)
{
// your stuff here
}
~DoUndoRAII(){
if(noexcept_){
// do whatever you need to do
}
}
void no_exception(){
noexcept_ = true;
}
private:
bool noexcept_;
};
void func(){
DoUndoRAII do_undo;
// last line
do_undo.no_exception();
}
When an exception is thrown, do_undo.no_exception()
will never be called and thus never set the noexcept_
value to true. :) An example can be found here on Ideone.
Upvotes: 1
Reputation: 23624
Let's assume that your F returns some class Helper:
Helper F()
{
MyClass doUndoWrapper;
}
When flow is normal - Helper is created. When exception is raised no copy of Helper is created. Try use this semantic by placing to private region constructor of Helper and declaring F as friend - so no one could create helper.
class Helper
{
private:
friend Helper F();
Helper(){ //place there OpDo semantic - first entry
// construct this class
Helper(const Helper& copy){ //this must present to allow stack operations
// copy constructor will be called at end of `F` to return value
// so place OpUndo semantic there to mark success without exception
Upvotes: 0
Reputation: 57046
Most people have used std::uncaught_exception()
to try to tell if an exception is pending, so they can throw an exception from a destructor if there isn't one already. That is generally considered Not A Good Idea.
If you want to not undo an operation if an exception has thrown, it should do the trick.
Remember that the destructor is your last chance to release any resources an object has, because after the destructor ends the object does not exist, and any resources it held are now permanently leaked. If OpDo()
allocates any memory or file handles or whatever, you do need to deal with that in the destructor no matter what.
Upvotes: 6