Re: Are throwing default constructors bad style, and if so, why?
on Sat Sep 20 2008, Andrei Alexandrescu <SeeWebsiteForEmail-AT-erdani.org> wrote:
I'm highly surprised by this behavior of deque. I'd have hoped that STL
containers at least abide to a no-throwing-default-construction
mantra.
The exception guarantees given by the standard are pretty much the
minimal set required in order to get useful work done. See
http://www.boost.org/community/exception_safety.html#footnote13
The way that that deque is defined, it's impossible to conservatively
acquire the contents of one deque.
(When I say "conservative" in this context I mean it as an analogy to
"energy conservative" in physics, i.e. no effort is expended. Swapping
and moving are conservative, copying is not.)
And what do you mean by "acquire?" Seems to me I can move from one
deque into another; that's conservative.
For example, IMHO move construction should never, ever throw. I think
this is an immutable ideal, just like the energy conservation principle
is in physics. We start with a number of objects. We end with the same
number of objects, just in a different place. There shouldn't be at any
point any "effort" expended on allocating extra memory, aside from the
"friction" of moving bits around. Making the default constructor throw
makes moving via swap (the only move available for C++03)
non-conservative
Hardly. First of all, swap doesn't use the default constructor at all.
Secondly, in C++03, swap should be individually defined for all
user-defined types, conservatively whenever possible. Third, in C++0x
there's a generic swap that is written in terms of move, which should be
conservative.
and consequently wasteful and potentially incorrect (if
peer code expects moving cannot engender exceptions). It is reasonable
to think that an object could embed a deque inside and, without risking
an exception, "absorbs" another deque.
That's just move semantics AFAICT, which should be nonthrowing. If you
mean something else, please explain.
Heck, if deque's constructor attempts to allocate memory (and
consequently throw), you can't even use the swap trick to ensure you
have completely emptied a deque.
Of course you can. It just might throw an exception. So what?
So picture the irony: _emptying_ a deque via the swap trick could
_throw_.
Sorry, I'm not getting much of a picture. Your swap trick looks like:
deque<int>().swap(d);
Plain as day you are constructing an object whose constructor may throw
an exception. Personally I'm more fond of
d.clear();
for that purpose.
This is a correctness issue that I guess simply rules the swap trick
out as a method of emptying a container in C++03.
I can't imagine why you'd say that.
--
Dave Abrahams
BoostPro Computing
http://www.boostpro.com
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]