When is a function try block useful?
Solution 1:
You use it in constructors to catch errors from initializers. Usually, you don't catch those errors, so this is a quite exceptional use.
Otherwise, it is useless: unless I'm proven wrong,
void f() try { ... } catch (...) { ... }
is strictly equivalent to
void f() { try { ... } catch (...) { ... } }
Solution 2:
Function try block are useful for me in two contexts.
a) To have a catch all clause around main()
allowing to write small utilities without having to worry about local error handling:
int main()
try {
// ...
return 0;
}
catch (...) {
// handle errors
return -1;
}
which is clearly just syntactic sugar for having a try/catch inside main()
itself.
b) to handle exceptions thrown by base class constructors:
struct B {
B() { /*might throw*/ }
};
struct A : B {
A()
try : B() {
// ...
}
catch (...) {
// handle exceptions thrown from inside A() or by B()
}
};
Solution 3:
Aside from the functional uses mentioned, you can use the function-try-block to save yourself one level of indentation. (Ack, an answer about coding styles!)
Typically you see examples with the function-try-block like so:
void f(/*...*/)
try {
/*...*/
}
catch(/*...*/) {
/*...*/
}
Where the function scope is indented to the same level as if there were no function-try-block. This can be useful when:
- you have an 80 character column limit and would have to wrap lines given the extra indentation.
- you are trying to retrofit some existing function with try catch and don't want to touch all the lines of the function. (Yeah, we could just use
git blame -w
.)
Though, for functions that are entirely wrapped with a function-try-block, I would suggest not alternating between some functions using function-try-blocks and some not within the same code base. Consistency is probably more important then line wrapping issues. :)
Solution 4:
Notes regarding how function try blocks operate:
For constructors, a function try block encompasses the construction of data members and base-classes.
For destructors, a function try block encompasses the destruction of data members and base-classes. It gets complicated, but for C++11, you have to include
noexcept(false)
in the declaration of your destructor (or that of a base/member class) or any destruction exception will result in termination at the conclusion of the catch block. It may be possible to prevent this by putting areturn
statement in the catch block (but this won't work for constructors).A catch block in a constructor or destructor must throw some exception (or it will implicitly rethrow the caught exception). It is not legal to simply
return
(at least in constructor's function catch block). Note, however, that you could callexit()
or similar, which might make sense in some situations.A catch block can't return a value, so it doesn't work for functions returning non-void (unless they intentionally terminate the program with
exit()
or similar). At least that is what I've read.The catch block for a constructor-function-try can't reference data/base members since they will have either have 1) not been constructed or 2) been destructed prior to the catch. As such, function try blocks are not useful for cleaning up an object's internal state -- the object should already be completely "dead" by the time you get there. This fact makes it very dangerous to use function try blocks in constructors, since it is difficult to police this rule over time if your compiler(s) don't happen to flag it.
valid (legal) uses
- Translating an exception (to a different type/message) thrown during the constructor or it's base/member constructors.
- Translating or absorbing and exception thrown during the destructor or it's base/member destructors (destructor etiquette notwithstanding).
- Terminating a program (perhaps with a useful message).
- Some kind of exception logging scheme.
- Syntactic sugar for void-returning functions that happen to need a fully encapsulating try/catch block.