C++20 has made it possible to default comparison operators, including the three-way comparison like this. <=>
can have a deduced return type, but other operators can't:
struct S {
friend auto operator<=>(const S&, const S&) = default; // OK
friend auto operator==(const S&, const S&) = default; // error, return type must be bool
auto& operator=(const S&) = default; // error, return type must be S&
};
Is there any reason why this restriction exists? It feels arbitrary.
The only return type that auto
can reasonably deduce to for operator==
is bool
.
Note: This is a wiki-style Q&A and its main goal is to raise awareness for restrictive language design and a c++26 proposal which aims to solve it.
The restriction is purely historical and there is a good chance that it's going to be lifted in C++26, assuming P2952R0
auto& operator=(X&&) = default
gets accepted into the standard.From the proposal:
In essence, the proposal would make deduced return types valid in those cases where a specific return type is expected. It would not alter the set of valid return types.
Implementation challenges and edge cases
It's worth noting that there is some compiler divergence when it comes to the set of valid return types. For example:
However, such divergence shouldn't stand in the way of the proposal because it proposes the "canonical" return type: