Re: Descriptive exceptions
* Eugene Gershnik:
Alf P. Steinbach wrote:
* Stefan Chrobot:
Some time ago I wrote a one-header class implementing Java-like
exception system.
It's generally not a good idea to add too much information to
exceptions.
It is not a good idea to include any human readable 'descriptions' but it is
a very good idea to include as much internal information as you can.
My reasoning does not appear in the quote, so this looks like some glib
unreasoned viewpoint exposed as such by contrasting it with another.
I did not comment directly on human readable descriptions carried by
exceptions, but to answer that first: it's not a good idea to think it's
not a good idea to use them; the what() function is there for a purpose,
/but/, there are good ways to use it, and bad ways (see para below).
Why it's a good idea to have a human readable description: it's
sometimes useful and the standard functionality provides for it. Why
it's a good idea to have that description in English: exception-carried
messages are meant for the programmer, not the user, and supporting
national languages in standard C++ exceptions is a nightmare. Probably
what you meant, or someone else (could even be me) meant and is
incorrectly conveyed above, is that it's generally not a good idea to
present exception-carried messages in the user interface, to base the
user interface on such carried messages, except possibly as additional
technical information. That's because of the internationalization
issues, because such carried messages are invariably technical, meant
for programmers and referring to internals of the system, and because a
message generated at the throw point very seldom can take into account
the context of the UI level. But even that rule has exceptions. ;-)
Now for the "as much internal information as you can" in the exception
itself. No, that isn't a good idea, and that's because of the reasons
snipped above: all that information is in most cases baggage of very
negative worth. It not only slows things down to collect it and copy
it, but collecting it and copying it can introduce additional failure
points and failure modes (as I illustrated with a concrete example),
which then invoke the exception-within-exception troll, and it
complicates the system, which has the same effect plus some.
KISS - Keep It Simple, Stupid.
mostly, except for logging at the point of throw,
Logging at the point of throw needlessly spends CPU/IO and adds noise to
logs even when exception is benign and handled graciously. More importantly
it requires each and every component in your system to be aware of whatever
logging facilities you use. Which doesn't work well with 3rd party libraries
or multi-component projects in general.
Sorry, to me that's not a meaningful statement (no meaning extracted).
On the fundamental level, by deciding to log before throwing, your code
proclaims that even though it has no clue how the condition has to be
handled or whether it is an error at all (hence the throw) it is absolutely
certain that it must be logged. Which may be fine for main() but is a little
presumtious of some utility code down the bottom of the call stack. ;-)
I think the implication of that paragraph is "fine for small toy
programs, but...". However, referring to the "call stack" for the
opposite kind of system, a less than minimal system, is less than
meaningful. The code throwing may execute in another process or on
another machine than the code finally receiving the exception.
The point of logging is to have a trace what happened, no matter what
happens later.
It may happen, for example, that an exception leads to catastrophic
failure (all the more likely when it's outfitted with a lot of data
baggage).
At the point of throw it's not known what may happen later.
Hence at that point, the code must act as if anything can happen.
[typically] what one's interested in in the
program code is just whether an exception occurred or not, and when
one's interested in the call stack and so on, one's typically running
the program in a debugger.
Typically?
Yes.
In my world typically you don't get to run debuggers on your
customers boxes and all you can count on are logs.
Yes, and note that (1) that isn't a contradiction, and (2) that it
really means, log if you can, at throw points. :-)
If the exception is one
of those "unexpected" ones that are handled by unwind-log-and-exit then you
really need to be able to log all you can about where and why it occured.
Yes, yes, yes.
Second only very simple applications are interested only in whether an
exception had occured because they treat all exceptions as fatal errors.
Sorry, to me that's not a meaningful statement (no meaning extracted).
On
a headless server you do want to reauthenticate if your ticket expired in
the middle of transaction, reconnect if the connection was lost and in
general handle as many things as you can without requiring user
intervention. So I usually have to look into additional error codes or other
information some exceptions carry. The nightmare scenario is when you have
to distinguish between two different variants of an error but the library
author chose to report both with a generic 'things are bad' exception.
Not sure what that is meant to pertain to.
--
A: Because it messes up the order in which people normally read text.
Q: Why is it such a bad thing?
A: Top-posting.
Q: What is the most annoying thing on usenet and in e-mail?
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]