Re: auto generated (default) constructora and copy operators
Dave Harris wrote:
Yes. But why would anyone do that? It's no better logically, and if the
default operator is inline there shouldn't be any performance difference.
There is no existing code that specifies default for == either. It doesn't
seem to be worth worrying about.
With your approach, if I want to define one of == and != I have to define
both. With my approach, I can get away with defining just one of them so
long as it is ==. So mine is a net win. It's more useful.
I do see this as being more about usefulness than correctness. Sometimes
the default definitions will be wrong, and then it will be a mistake to
request them, but I see that as the programmer's fault rather than the
language's. It's no different to writing the wrong definition out by hand.
The programmer remains responsible for doing the right thing - you can't
avoid that.
I'm getting a little confused here. If the equality and inequality
operators were compiler-generated without a request, we would save the
need to declare one or both. However, since we are talking about where
the auto/default versions must be explicitly requested, the difference
in the number of keystrokes between writing "auto" (or maybe "{
default; }") and "{ return !(*this==B); }" is minimal (the definition
of == coincides between the two suggestions). I do not call that
significantly useful.
The main problem is that == and != may be defined in different translation
units, so the compiler can't look at == when deciding how to implement !=
and vice versa.
If it weren't for that, I'd be happy to say that default == is only
permitted when != is also default. But it doesn't really matter, and in
practice implementations will usually be able to detect the situation and
issue a warning.
I'm afraid I must be missing something here. If == is inline
(including auto on the declaration in the header) there is no problem,
and optimal versions of both can be generated. Whether == is defined
in the same or a different translation unit, != must call == (since it
is not inline), whether it is auto or not. To put it differently, I do
not see any semantic or implementation difference between "auto" and
"return !(*this==B).
James Dennett wrote:
Manfred von Willich wrote:
If the rule is as you suggested it, that
<snip>
then if we write
bool operator==(thisclass const & B) auto;
bool operator!=(thisclass const & B) const { return nonsense(); }
then the operators == and != may be inconsistent.
Not very relevant, in my estimation: it doesn't make
it impossible to be inconsistent, but it does mean
that natural code will be consistent.
Ergo, your suggestion does not necessarily avoid inconsistency between
== and !=, in apparently reasonable code.
The code you show above is unreasonable, in my book.
I've never had cause to implement operator!= as
anything other than the negation of ==.
The benefit of being able to
say
bool operator!=(thisclass const &) const auto;
instead of
bool operator!=(thisclass const & B) const { return !operator==(B);
}
is rather doubtful, especially since the latter is more immediately
obvious.
We can disagree there.
See my response to Dave above. I presume that here you are referring
to what is more immediately obvious, not the value of the benefit,
which as I say is minimal.
C++ does not even pretend to encourage operator consistency - it lets
us hijack operators for unrelated uses (e.g. cout::operator<<).
C++ does encourage consistency, it just doesn't enforce
it at a language level (though the library requires a
certain level of consistency).
How so? At language level I think that aside from the
compiler-generated methods and the existing consistency with predefined
types, the apparently related operators are entirely unrelated. Be
careful that you don't read your bias into the language.
Even at library level (e.g. a sort template or a polymorphic base
class), the requirements on comparison operators provided by a user
class are generally made very explicit, including exactly which
operators are used. If a library template or polymorphic base class
uses both == and != (at least without catering specifically for
inconsistent semantics), such a library should be rewritten using only
one and the original author banished. Where library classes intend to
use an intuitive meaning for most operators, this consistency has been
built in. I do not read this as encouragement by C++ - only standard
style of usage.
Manfred
---
[ comp.std.c++ is moderated. To submit articles, try just posting with ]
[ your news-reader. If that fails, use mailto:std-c++@ncar.ucar.edu ]
[ --- Please see the FAQ before posting. --- ]
[ FAQ: http://www.comeaucomputing.com/csc/faq.html ]