Created on 2019-12-18.00:00:00 last changed 58 months ago
Proposed resolution:
This wording is relative to N4842.
Modify [cmp.concept] as indicated:
template<class T, class U, class Cat = partial_ordering>
concept three_way_comparable_with =
weakly-equality-comparable-with<T, U> &&
partially-ordered-with<T, U> &&
three_way_comparable<T, Cat> &&
three_way_comparable<U, Cat> &&
common_reference_with<const remove_reference_t<T>&, const remove_reference_t<U>&> &&
three_way_comparable<
common_reference_t<const remove_reference_t<T>&, const remove_reference_t<U>&>, Cat> &&
weakly-equality-comparable-with<T, U> &&
partially-ordered-with<T, U> &&
requires(const remove_reference_t<T>& t, const remove_reference_t<U>& u) {
{ t <=> u } -> compares-as<Cat>;
{ u <=> t } -> compares-as<Cat>;
};
[ 2020-01 Status set to Tentatively Ready after five positive votes on the reflector. ]
The concept three_way_comparable_with is defined in [cmp.concept] as:
template<class T, class U, class Cat = partial_ordering>
concept three_way_comparable_with =
weakly-equality-comparable-with<T, U> &&
partially-ordered-with<T, U> &&
three_way_comparable<T, Cat> &&
three_way_comparable<U, Cat> &&
common_reference_with<const remove_reference_t<T>&, const remove_reference_t<U>&> &&
three_way_comparable<
common_reference_t<const remove_reference_t<T>&, const remove_reference_t<U>&>, Cat> &&
requires(const remove_reference_t<T>& t, const remove_reference_t<U>& u) {
{ t <=> u } -> compares-as<Cat>;
{ u <=> t } -> compares-as<Cat>;
};
Which notably doesn't follow the requirement ordering:
same-type requirements on T
same-type requirements on U
common_reference_with requirement
same-type requirements on common_reference_t<T, U>
cross-type requirements on T and U
that the other cross-type comparison concepts ([concept.equalitycomparable], [concept.totallyordered]) use. There were some motivating reasons for that ordering:
The existence of a common reference type is effectively an opt-in to cross-type concepts. Avoiding checking cross-type expressions when no common reference type exists can enable the concepts to work even in the presence of poorly-constrained "accidental" cross-type operator templates which could otherwise produce compile errors instead of dissatisfied concepts.
Putting the simpler same-type requirements first can help produce simpler error messages when applying the wrong concept to a pair of types, or the right concept to the wrong pair of types. "Frobnozzle <=> Frobnozzle is not a valid expression" is more easily deciphered than is "std::common_reference<int, FrobNozzle> has no member named type".
three_way_comparable_with should be made consistent with equality_comparable_with and totally_ordered_with for the above reasons and to make it easier to reason about comparison concepts in general.
| History | |||
|---|---|---|---|
| Date | User | Action | Args |
| 2021-02-25 10:48:01 | admin | set | status: wp -> c++20 |
| 2020-02-24 16:02:59 | admin | set | status: voting -> wp |
| 2020-01-17 04:54:50 | admin | set | status: ready -> voting |
| 2020-01-05 16:13:01 | admin | set | messages: + msg10910 |
| 2020-01-05 16:13:01 | admin | set | status: new -> ready |
| 2019-12-21 14:36:49 | admin | set | messages: + msg10899 |
| 2019-12-18 00:00:00 | admin | create | |