The Mothership Has Landed
Adding <=>
to the Library
The work of integrating operator<=>
into the library has been performed by multiple different papers, each addressing a different aspect of the integration. In the interest of streamlining review by the Library Working Group, the wording has been combined into a single paper. This is that paper.
In San Diego and Kona, several papers were approved by LEWG adding functionality to the library related to comparisons. What follows is the list of those papers, in alphabetical order, with a brief description of what those papers are. The complete motivation and design rationale for each can be found within the papers themselves.
operator<=>
to the standard library types whose behavior is not dependent on a template parameter.XXX_order
algorithms customization points and introducing compare_XXX_order_fallback
algorithms that preferentially invoke the former algorithm and fallback to synthesizing an ordering from ==
and <
(using the rules from P1186R1).has_strong_structural_equality<T>
(useful to check if a type can be used as a non-type template parameter).compare_three_way_result<T>
, the concepts ThreeWayComparable<T>
and ThreeWayComparableWith<T,U>
, removing the algorithm compare_3way
and replacing it with a function comparison object compare_three_way
(i.e. the <=>
version of std::ranges::less
).operator<=>
to the standard library types whose behavior is dependent on a template parameter, removing those equality operators made redundant by P1185R1 and defaulting operator==
where appropriate.common_type
for the comparison categories.strong_order
and weak_order
.LEWG's unanimous preference was that operator<=>
s be declared as hidden friends.
This initial draft is complete up through Clause 25 and includes coroutine_handle<>
. No work has yet been done on Clauses 26-31.
There are a few things that will change behavior as a result of all these papers and the chosen direction for declaring operators as hidden friends.
For the preexisting non-member, non-template comparison operators, any comparison that relies on finding the operator in std
with regular unqualified lookup will fail:
using namespace std;
struct X { operator error_code() const; };
X{} == X{}; // ok in C++17, ill-formed with this change
X{} == error_code{}; // ok
Here is a more subtle example, reproduced from the LLVM codebase:
struct StringRef {
StringRef(std::string const&); // NB: non-explicit
operator std::string() const; // NB: non-explicit
};
bool operator==(StringRef, StringRef);
bool f(StringRef a, std::string b) {
return a == b; // (*)
}
In C++17, the marked line is well-formed. The operator==
for basic_string
is a non-member function template, and so would not be considered a candidate; the only viable candidate is the operator==
taking two StringRef
s. With the proposed changes, the operator==
for basic_string
becomes a non-member hidden friend, non-template, which makes it a candidate (converting a
to a string
). That candidate is ambiguous with the operator==(StringRef, StringRef)
candidate - each requires a conversion in one argument, so the call becomes ill-formed.
bool is42(std::variant<int, std::string> const& v) {
return v == 42; // (*)
}
In C++17, the operator==
for variant
is a non-member function template and is thus not a viable candidate for the marked line. That check is ill-formed. With the proposed changes, the operator==
for variant
becomes a non-member hidden friend, non-template, which makes it a candidate (converting 42
to a variant<int, string>
). This is arguably a fix, since both variant<int, string> v = 42;
and v = 42;
are already well-formed, so it is surely reasonable that v == 42
is as well.
Thank you to Casey Carter for the tremendous wording review.
Change 15.4.2.1/2 [expos.only.func]:
The following
function isare defined for exposition only to aid in the specification of the library:
and append:
constexpr auto
synth-3way
= []<class T, class U>(const T& t, const U& u) requires requires { { t < u } -> bool; { u < t } -> bool; } { if constexpr (ThreeWayComparableWith<T, U>) { return t <=> u; } else { if (t < u) return weak_ordering::less; if (u < t) return weak_ordering::greater; return weak_ordering::equivalent; } }; template<class T, class U=T> using
synth-3way-result
= decltype(
synth-3way
(declval<T&>(), declval<U&>()));
Remove 15.4.2.3 [operators], which begins:
In this library, whenever a declaration is provided for anoperator!=
,operator>
,operator<=
, oroperator>=
for a typeT
, its requirements and semantics are as follows, unless explicitly specified otherwise.
Add a clause to 15.5.5 [conforming], probably after 15.5.5.4 [global.functions]. Not strictly related to <=>
as a whole, but it's a requirement that's currently missing and needs to be added somewhere. See also P1601.
15.5.5.x Hidden friend functions [conforming.hidden.friend]
An implementation shall not provide any additional out-of-class declarations or redeclarations for any non-member function specified as a non-member
friend
and defined within the body of a class. [ Note: The intent is that such functions are to be found via argument-dependent lookup only. -end note ]
Added: compare_three_way_result
, concepts ThreeWayComparable
and ThreeWayComparableWith
, compare_three_way
and compare_XXX_order_fallback
Changed operators for: type_info
Respecified: strong_order()
, weak_order()
, and partial_order()
Removed: compare_3way()
, strong_equal()
, and weak_equal()
In 16.7.2 [type.info], remove operator!=
:
namespace std { class type_info { public: virtual ~type_info(); bool operator==(const type_info& rhs) const noexcept;
bool operator!=(const type_info& rhs) const noexcept;bool before(const type_info& rhs) const noexcept; size_t hash_code() const noexcept; const char* name() const noexcept; type_info(const type_info& rhs) = delete; // cannot be copied type_info& operator=(const type_info& rhs) = delete; // cannot be copied }; }
and
Effects: Compares the current object with rhs.bool operator==(const type_info& rhs) const noexcept;
Returns:true
if the two values describe the same type.
bool operator!=(const type_info& rhs) const noexcept;Returns:!(*this == rhs)
.
Add into 16.11.1 [compare.syn]:
namespace std { // [cmp.categories], comparison category types class weak_equality; class strong_equality; class partial_ordering; class weak_ordering; class strong_ordering; // named comparison functions constexpr bool is_eq (weak_equality cmp) noexcept { return cmp == 0; } constexpr bool is_neq (weak_equality cmp) noexcept { return cmp != 0; } constexpr bool is_lt (partial_ordering cmp) noexcept { return cmp < 0; } constexpr bool is_lteq(partial_ordering cmp) noexcept { return cmp <= 0; } constexpr bool is_gt (partial_ordering cmp) noexcept { return cmp > 0; } constexpr bool is_gteq(partial_ordering cmp) noexcept { return cmp >= 0; } // common_type specializations template<> struct common_type<strong_equality, partial_ordering> { using type = weak_equality; }; template<> struct common_type<partial_ordering, strong_equality> { using type = weak_equality; }; template<> struct common_type<strong_equality, weak_ordering> { using type = weak_equality; }; template<> struct common_type<weak_ordering, strong_equality> { using type = weak_equality; }; // [cmp.common], common comparison category type template<class... Ts> struct common_comparison_category { using type = see below; }; template<class... Ts> using common_comparison_category_t = typename common_comparison_category<Ts...>::type; // [cmp.concept], concept ThreeWayComparable template<class T, class Cat = partial_ordering> concept ThreeWayComparable = see below; template<class T, class U, class Cat = partial_ordering> concept ThreeWayComparableWith = see below; // [cmp.result], spaceship invocation result template<class T, class U = T> struct compare_three_way_result; template<class T, class U = T> using compare_three_way_result_t = typename compare_three_way_result<T, U>::type; // [cmp.object], spaceship object struct compare_three_way; // [cmp.alg], comparison algorithms
template<class T> constexpr strong_ordering strong_order(const T& a, const T& b);template<class T> constexpr weak_ordering weak_order(const T& a, const T& b);template<class T> constexpr partial_ordering partial_order(const T& a, const T& b);template<class T> constexpr strong_equality strong_equal(const T& a, const T& b);template<class T> constexpr weak_equality weak_equal(const T& a, const T& b);inline namespace unspecified { inline constexpr unspecified strong_order = unspecified; inline constexpr unspecified weak_order = unspecified; inline constexpr unspecified partial_order = unspecified; inline constexpr unspecified compare_strong_order_fallback = unspecified; inline constexpr unspecified compare_weak_order_fallback = unspecified; inline constexpr unspecified compare_partial_order_fallback = unspecified; } }
Change 16.11.2.2 [cmp.weakeq]:
namespace std { class weak_equality { int value; // exposition only [...] // comparisons friend constexpr bool operator==(weak_equality v, unspecified) noexcept
;{ return v.value == 0; }friend constexpr bool operator!=(weak_equality v, unspecified) noexcept;friend constexpr bool operator==(unspecified, weak_equality v) noexcept;friend constexpr bool operator!=(unspecified, weak_equality v) noexcept;friend constexpr bool operator==(weak_equality v, weak_equality w) noexcept = default; friend constexpr weak_equality operator<=>(weak_equality v, unspecified) noexcept;{ return v; } friend constexpr weak_equality operator<=>(unspecified, weak_equality v) noexcept;{ return v; } }; // valid values' definitions inline constexpr weak_equality weak_equality::equivalent(eq::equivalent); inline constexpr weak_equality weak_equality::nonequivalent(eq::nonequivalent); }
Remove the rest of the clause (now defined inline):
constexpr bool operator==(weak_equality v, unspecified) noexcept; constexpr bool operator==(unspecified, weak_equality v) noexcept;Returns:v.value == 0
.
constexpr bool operator!=(weak_equality v, unspecified) noexcept; constexpr bool operator!=(unspecified, weak_equality v) noexcept;Returns:v.value != 0
.
constexpr weak_equality operator<=>(weak_equality v, unspecified) noexcept; constexpr weak_equality operator<=>(unspecified, weak_equality v) noexcept;Returns:v
.
Change 16.11.2.3 [cmp.strongeq]:
namespace std { class strong_equality { int value; // exposition only [...] // comparisons friend constexpr bool operator==(strong_equality v, unspecified) noexcept
;{ return v.value == 0; }friend constexpr bool operator!=(strong_equality v, unspecified) noexcept;friend constexpr bool operator==(unspecified, strong_equality v) noexcept;friend constexpr bool operator!=(unspecified, strong_equality v) noexcept;friend constexpr bool operator==(strong_equality v, strong_equality w) noexcept = default; friend constexpr strong_equality operator<=>(strong_equality v, unspecified) noexcept;{ return v; } friend constexpr strong_equality operator<=>(unspecified, strong_equality v) noexcept;{ return v; } }; // valid values' definitions inline constexpr strong_equality strong_equality::equal(eq::equal); inline constexpr strong_equality strong_equality::nonequal(eq::nonequal); inline constexpr strong_equality strong_equality::equivalent(eq::equivalent); inline constexpr strong_equality strong_equality::nonequivalent(eq::nonequivalent); }
Remove most of the rest of the clause:
Returns:constexpr operator weak_equality() const noexcept;
value == 0 ? weak_equality::equivalent : weak_equality::nonequivalent
.
constexpr bool operator==(strong_equality v, unspecified) noexcept; constexpr bool operator==(unspecified, strong_equality v) noexcept;Returns:v.value == 0
.
constexpr bool operator!=(strong_equality v, unspecified) noexcept; constexpr bool operator!=(unspecified, strong_equality v) noexcept;Returns:v.value != 0
.
constexpr strong_equality operator<=>(strong_equality v, unspecified) noexcept; constexpr strong_equality operator<=>(unspecified, strong_equality v) noexcept;Returns:v
.
Change 16.11.2.4 [cmp.partialord]:
namespace std { class partial_ordering { int value; // exposition only bool is_ordered; // exposition only [...] // conversion constexpr operator weak_equality() const noexcept; // comparisons friend constexpr bool operator==(partial_ordering v, unspecified) noexcept;
friend constexpr bool operator!=(partial_ordering v, unspecified) noexcept;friend constexpr bool operator==(partial_ordering v, partial_ordering w) noexcept = default; friend constexpr bool operator< (partial_ordering v, unspecified) noexcept; friend constexpr bool operator> (partial_ordering v, unspecified) noexcept; friend constexpr bool operator<=(partial_ordering v, unspecified) noexcept; friend constexpr bool operator>=(partial_ordering v, unspecified) noexcept;friend constexpr bool operator==(unspecified, partial_ordering v) noexcept;friend constexpr bool operator!=(unspecified, partial_ordering v) noexcept;friend constexpr bool operator< (unspecified, partial_ordering v) noexcept; friend constexpr bool operator> (unspecified, partial_ordering v) noexcept; friend constexpr bool operator<=(unspecified, partial_ordering v) noexcept; friend constexpr bool operator>=(unspecified, partial_ordering v) noexcept; friend constexpr partial_ordering operator<=>(partial_ordering v, unspecified) noexcept; friend constexpr partial_ordering operator<=>(unspecified, partial_ordering v) noexcept; }; [...] }
Remove just the extra ==
and !=
operators in 16.11.2.4 [cmp.partialord]/3 and 4:
Returns: Forconstexpr bool operator==(partial_ordering v, unspecified) noexcept; constexpr bool operator< (partial_ordering v, unspecified) noexcept; constexpr bool operator> (partial_ordering v, unspecified) noexcept; constexpr bool operator<=(partial_ordering v, unspecified) noexcept; constexpr bool operator>=(partial_ordering v, unspecified) noexcept;
operator@
,v.is_ordered && v.value @ 0
.Returns: For
constexpr bool operator==(unspecified, partial_ordering v) noexcept;constexpr bool operator< (unspecified, partial_ordering v) noexcept; constexpr bool operator> (unspecified, partial_ordering v) noexcept; constexpr bool operator<=(unspecified, partial_ordering v) noexcept; constexpr bool operator>=(unspecified, partial_ordering v) noexcept;operator@
,v.is_ordered && 0 @ v.value
.
constexpr bool operator!=(partial_ordering v, unspecified) noexcept; constexpr bool operator!=(unspecified, partial_ordering v) noexcept;Returns: Foroperator@
,!v.is_ordered || v.value != 0
.
Change 16.11.2.5 [cmp.weakord]:
namespace std { class weak_ordering { int value; // exposition only [...] // comparisons friend constexpr bool operator==(weak_ordering v, unspecified) noexcept; friend constexpr bool operator==(weak_ordering v, weak_ordering w) noexcept = default;
friend constexpr bool operator!=(weak_ordering v, unspecified) noexcept;friend constexpr bool operator< (weak_ordering v, unspecified) noexcept; friend constexpr bool operator> (weak_ordering v, unspecified) noexcept; friend constexpr bool operator<=(weak_ordering v, unspecified) noexcept; friend constexpr bool operator>=(weak_ordering v, unspecified) noexcept;friend constexpr bool operator==(unspecified, weak_ordering v) noexcept;friend constexpr bool operator!=(unspecified, weak_ordering v) noexcept;friend constexpr bool operator< (unspecified, weak_ordering v) noexcept; friend constexpr bool operator> (unspecified, weak_ordering v) noexcept; friend constexpr bool operator<=(unspecified, weak_ordering v) noexcept; friend constexpr bool operator>=(unspecified, weak_ordering v) noexcept; friend constexpr weak_ordering operator<=>(weak_ordering v, unspecified) noexcept; friend constexpr weak_ordering operator<=>(unspecified, weak_ordering v) noexcept; }; [...] };
Remove just the extra ==
and !=
operators from 16.11.2.5 [cmp.weakord]/4 and /5:
Returns:constexpr bool operator==(weak_ordering v, unspecified) noexcept;
constexpr bool operator!=(weak_ordering v, unspecified) noexcept;constexpr bool operator< (weak_ordering v, unspecified) noexcept; constexpr bool operator> (weak_ordering v, unspecified) noexcept; constexpr bool operator<=(weak_ordering v, unspecified) noexcept; constexpr bool operator>=(weak_ordering v, unspecified) noexcept;v.value @ 0
foroperator@
.Returns:
constexpr bool operator==(unspecified, weak_ordering v) noexcept;constexpr bool operator!=(unspecified, weak_ordering v) noexcept;constexpr bool operator< (unspecified, weak_ordering v) noexcept; constexpr bool operator> (unspecified, weak_ordering v) noexcept; constexpr bool operator<=(unspecified, weak_ordering v) noexcept; constexpr bool operator>=(unspecified, weak_ordering v) noexcept;0 @ v.value
foroperator@
.
Change 16.11.2.6 [cmp.strongord]:
namespace std { class strong_ordering { int value; // exposition only [...] // comparisons friend constexpr bool operator==(strong_ordering v, unspecified) noexcept; friend constexpr bool operator==(strong_ordering v, strong_ordering w) noexcept = default;
friend constexpr bool operator!=(strong_ordering v, unspecified) noexcept;friend constexpr bool operator< (strong_ordering v, unspecified) noexcept; friend constexpr bool operator> (strong_ordering v, unspecified) noexcept; friend constexpr bool operator<=(strong_ordering v, unspecified) noexcept; friend constexpr bool operator>=(strong_ordering v, unspecified) noexcept;friend constexpr bool operator==(unspecified, strong_ordering v) noexcept;friend constexpr bool operator!=(unspecified, strong_ordering v) noexcept;friend constexpr bool operator< (unspecified, strong_ordering v) noexcept; friend constexpr bool operator> (unspecified, strong_ordering v) noexcept; friend constexpr bool operator<=(unspecified, strong_ordering v) noexcept; friend constexpr bool operator>=(unspecified, strong_ordering v) noexcept; friend constexpr strong_ordering operator<=>(strong_ordering v, unspecified) noexcept; friend constexpr strong_ordering operator<=>(unspecified, strong_ordering v) noexcept; }; [...] }
Remove just the extra ==
and !=
operators from 16.11.2.6 [cmp.strongord]/6 and /7:
Returns:constexpr bool operator==(strong_ordering v, unspecified) noexcept;
constexpr bool operator!=(strong_ordering v, unspecified) noexcept;constexpr bool operator< (strong_ordering v, unspecified) noexcept; constexpr bool operator> (strong_ordering v, unspecified) noexcept; constexpr bool operator<=(strong_ordering v, unspecified) noexcept; constexpr bool operator>=(strong_ordering v, unspecified) noexcept;v.value @ 0
foroperator@
.Returns:
constexpr bool operator==(unspecified, strong_ordering v) noexcept;constexpr bool operator!=(unspecified, strong_ordering v) noexcept;constexpr bool operator< (unspecified, strong_ordering v) noexcept; constexpr bool operator> (unspecified, strong_ordering v) noexcept; constexpr bool operator<=(unspecified, strong_ordering v) noexcept; constexpr bool operator>=(unspecified, strong_ordering v) noexcept;0 @ v.value
foroperator@
.
Add a new subclause [cmp.concept] "concept ThreeWayComparable
":
template <typename T, typename Cat> concept
compares-as
= // exposition only Same<common_comparison_category_t<T, Cat>, Cat>;
template<class T, class U> concept
partially-ordered-with
= // exposition only requires(const remove_reference_t<T>& t, const remove_reference_t<U>& u) { { t < u } -> Boolean; { t > u } -> Boolean; { t <= u } -> Boolean; { t >= u } -> Boolean; { u < t } -> Boolean; { u > t } -> Boolean; { u <= t } -> Boolean; { u >= t } -> Boolean;
};Let
t
andu
be lvalues of typesconst remove_reference_t<T>
andconst remove_reference_t<U>
respectively.partially-ordered-with<T, U>
is satisfied only if:
t < u
,t <= u
,t > u
,t >= u
,u < t
,u <= t
,u > t
, andu >= t
have the same domain.bool(t < u) == bool(u > t)
bool(u < t) == bool(t > u)
bool(t <= u) == bool(u >= t)
bool(u <= t) == bool(t >= u)
template <typename T, typename Cat = partial_ordering> concept ThreeWayComparable =
weakly-equality-comparable-with
<T, T> && (!ConvertibleTo<Cat, partial_ordering> ||
partially-ordered-with
<T, T>) && requires(const remove_reference_t<T>& a, const remove_reference_t<T>& b) { { a <=> b } ->
compares-as
<Cat>; };
Let
a
andb
be lvalues of typeconst remove_reference_t<T>
.T
andCat
modelThreeWayComparable<T, Cat>
only if:
(a <=> b == 0) == bool(a == b)
.(a <=> b != 0) == bool(a != b)
.((a <=> b) <=> 0)
and(0 <=> (b <=> a))
are equal- If
Cat
is convertible tostrong_equality
,T
modelsEqualityComparable
([concept.equalitycomparable]).- If
Cat
is convertible topartial_ordering
:
(a <=> b < 0) == bool(a < b)
.(a <=> b > 0) == bool(a > b)
.(a <=> b <= 0) == bool(a <= b)
.(a <=> b >= 0) == bool(a >= b)
.- If
Cat
is convertible tostrong_ordering
,T
modelsStrictTotallyOrdered
([concept.stricttotallyordered]).Lettemplate <typename T, typename U, typename Cat = partial_ordering> concept ThreeWayComparableWith =
weakly-equality-comparable-with
<T, U> && (!ConvertibleTo<Cat, partial_ordering> ||
partially-ordered-with
<T, U>) && ThreeWayComparable<T, Cat> && ThreeWayComparable<U, Cat> && CommonReference<const remove_reference_t<T>&, const remove_reference_t<U>&> && ThreeWayComparable< 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>; };
t
andu
be lvalues of typesconst remove_reference_t<T>
andconst remove_reference_t<U>
, respectively. LetC
becommon_reference_t<const remove_reference_t<T>&, const remove_reference_t<U>&>
.T
,U
, andCat
modelThreeWayComparableWith<T, U, Cat>
only if:
t <=> u
andu <=> t
have the same domain.((t <=> u) <=> 0)
and(0 <=> (u <=> t))
are equal(t <=> u == 0) == bool(t == u)
.(t <=> u != 0) == bool(t != u)
.Cat(t <=> u) == Cat(C(t) <=> C(u))
.- If
Cat
is convertible tostrong_equality
,T
andU
modelEqualityComparableWith<T, U>
([concepts.equalitycomparable]).- If
Cat
is convertible topartial_ordering
:
(t <=> u < 0) == bool(t < u)
(t <=> u > 0) == bool(t > u)
(t <=> u <= 0) == bool(t <= u)
(t <=> u >= 0) == bool(t >= u)
- If
Cat
is convertible tostrong_ordering
,T
andU
modelStrictTotallyOrderedWith<T, U>
([concepts.stricttotallyordered]).
Add a new subclause [cmp.result] "spaceship invocation result":
The behavior of a program that adds specializations for the
compare_three_way_result
template defined in this subclause is undefined.For the
compare_three_way_result
type trait applied to the typesT
andU
, lett
andu
denote lvalues of typesconst remove_reference_t<T>
andconst remove_reference_t<U>
, respectively. If the expressiont <=> u
is well-formed, the member typedef-nametype
denotes the typedecltype(t <=> u)
. Otherwise, there is no membertype
.
Add a new subclause [cmp.object] "spaceship object":
In this subclause,
BUILTIN_PTR_3WAY(T, U)
for typesT
andU
is a boolean constant expression.BUILTIN_PTR_3WAY(T, U)
istrue
if and only if<=>
in the expressiondeclval<T>() <=> declval<U>()
resolves to a built-in operator comparing pointers.struct compare_three_way { template<class T, class U> requires ThreeWayComparableWith<T,U> || BUILTIN_PTR_3WAY(T, U) constexpr auto operator()(T&& t, U&& u) const; using is_transparent = unspecified; };
Expects: If the expression
std::forward<T>(t) <=> std::forward<U>(u)
results in a call to a built-in operator<=>
comparing pointers of typeP
, the conversion sequences from bothT
andU
toP
are equality-preserving ([concepts.equality]).Effects:
- If the expression
std::forward<T>(t) <=> std::forward<U>(u)
results in a call to a built-in operator<=>
comparing pointers of typeP
: returnsstrong_ordering::less
if (the converted value of)t
precedesu
in the implementation-defined strict total order ([range.cmp]) over pointers of typeP
,strong_ordering::greater
ifu
precedest
, and otherwisestrong_ordering::equal
.- Otherwise, equivalent to:
return std::forward<T>(t) <=> std::forward<U>(u);
In addition to being available via inclusion of the
<compare>
header, the classcompare_three_way
is available when the header<functional>
is included.
Replace the entirety of 16.11.4 [cmp.alg]. This wording relies on the specification-only function 3WAY<R>
defined in P1186R1.
template<class T> constexpr strong_ordering strong_order(const T& a, const T& b);Effects: Compares two values and produces a result of typestrong_ordering
:
If numeric_limits::is_iec559 is true, returns a result of type strong_ordering that is consistent with the totalOrder operation as specified in ISO/IEC/IEEE 60559. Otherwise, returns a <=> b if that expression is well-formed and convertible to strong_ordering.Otherwise, if the expression a <=> b is well-formed, then the function is defined as deleted.Otherwise, if the expressions a == b and a < b are each well-formed and convertible to bool, then
if a == b is true, returns strong_ordering::equal;otherwise, if a < b is true, returns strong_ordering::less;otherwise, returns strong_ordering::greater.Otherwise, the function is defined as deleted.
template<class T> constexpr weak_ordering weak_order(const T& a, const T& b);Effects: Compares two values and produces a result of type weak_ordering:
Returns a <=> b if that expression is well-formed and convertible to weak_ordering.Otherwise, if the expression a <=> b is well-formed, then the function is defined as deleted.Otherwise, if the expressions a == b and a < b are each well-formed and convertible to bool, then
if a == b is true, returns weak_ordering::equivalent;otherwise, if a < b is true, returns weak_ordering::less;otherwise, returns weak_ordering::greater.Otherwise, the function is defined as deleted.
template<class T> constexpr partial_ordering partial_order(const T& a, const T& b);Effects: Compares two values and produces a result of type partial_ordering:
Returns a <=> b if that expression is well-formed and convertible to partial_ordering.Otherwise, if the expression a <=> b is well-formed, then the function is defined as deleted.Otherwise, if the expressions a == b and a < b are each well-formed and convertible to bool, then
if a == b is true, returns partial_ordering::equivalent;otherwise, if a < b is true, returns partial_ordering::less;otherwise, returns partial_ordering::greater.Otherwise, the function is defined as deleted.
template<class T> constexpr strong_equality strong_equal(const T& a, const T& b);Effects: Compares two values and produces a result of type strong_equality:
Returns a <=> b if that expression is well-formed and convertible to strong_equality.Otherwise, if the expression a <=> b is well-formed, then the function is defined as deleted.Otherwise, if the expression a == b is well-formed and convertible to bool, then
if a == b is true, returns strong_equality::equal;otherwise, returns strong_equality::nonequal.Otherwise, the function is defined as deleted.
template<class T> constexpr weak_equality weak_equal(const T& a, const T& b);Effects: Compares two values and produces a result of type weak_equality:
Returns a <=> b if that expression is well-formed and convertible to weak_equality.Otherwise, if the expression a <=> b is well-formed, then the function is defined as deleted.Otherwise, if the expression a == b is well-formed and convertible to bool, then
if a == b is true, returns weak_equality::equivalent;otherwise, returns weak_equality::nonequivalent.Otherwise, the function is defined as deleted.The name
strong_order
denotes a customization point object ([customization.point.object]). The expressionstrong_order(E, F)
for some subexpressionsE
andF
is expression-equivalent ([defns.expression-equivalent]) to the following:
- If the decayed types of
E
andF
differ,strong_order(E, F)
is ill-formed.- Otherwise,
strong_ordering(strong_order(E, F))
if it is a well-formed expression with overload resolution performed in a context that does not include a declaration ofstd::strong_order
.- Otherwise, if the decayed type
T
ofE
andF
is a floating point type, yields a value of typestrong_ordering
that is consistent with the ordering observed byT
's comparison operators, and ifnumeric_limits<T>::is_iec559
istrue
is additionally consistent with the totalOrder operation as specified in ISO/IEC/IEEE 60599.- Otherwise,
strong_ordering(E <=> F)
if it is a well-formed expression.- Otherwise,
strong_order(E, F)
is ill-formed. [Note: This case can result in substitution failure whenstrong_order(E, F)
appears in the immediate context of a template instantiation. —end note]The name
weak_order
denotes a customization point object ([customization.point.object]). The expressionweak_order(E, F)
for some subexpressionsE
andF
is expression-equivalent ([defns.expression-equivalent]) to the following:
- If the decayed types of
E
andF
differ,weak_order(E, F)
is ill-formed.- Otherwise,
weak_ordering(weak_order(E, F))
if it is a well-formed expression with overload resolution performed in a context that does not include a declaration ofstd::weak_order
.- Otherwise, if the decayed type
T
ofE
andF
is a floating point type, yields a value of typeweak_ordering
that is consistent with the ordering observed byT
's comparison operators andstrong_order
, and ifnumeric_liits<T>::is_iec559
istrue
is additionally consistent with the following equivalence classes, ordered from lesser to greater:
- Together, all negative NaN values
- Negative infinity
- Each normal negative value
- Each subnormal negative value
- Together, both zero values
- Each subnormal positive value
- Each normal positive value
- Positive infinity
- Together, all positive NaN values
- Otherwise,
weak_ordering(strong_order(E, F))
if it is a well-formed expression.- Otherwise,
weak_ordering(E <=> F)
if it is a well-formed expression.- Otherwise,
weak_order(E, F)
is ill-formed. [Note: This case can result in substitution failure whenstd::weak_order(E, F)
appears in the immediate context of a template instantiation. —end note]The name
partial_order
denotes a customization point object ([customization.point.object]). The expressionpartial_order(E, F)
for some subexpressionsE
andF
is expression-equivalent ([defns.expression-equivalent]) to the following:
- If the decayed types of
E
andF
differ,partial_order(E, F)
is ill-formed.- Otherwise,
partial_ordering(partial_order(E, F))
if it is a well-formed expression with overload resolution performed in a context that does not include a declaration ofstd::partial_order
.- Otherwise,
partial_ordering(weak_order(E, F))
if it is a well-formed expression.- Otherwise,
partial_ordering(E <=> F)
if it is a well-formed expression.- Otherwise,
partial_order(E, F)
is ill-formed. [Note: This case can result in substitution failure whenstd::partial_order(E, F)
appears in the immediate context of a template instantiation. —end note]The name
compare_strong_order_fallback
denotes a comparison customization point ([customization.point.object]) object. The expressioncompare_strong_order_fallback(E, F)
for some subexpressionsE
andF
is expression-equivalent ([defns.expression-equivalent]) to:
- If the decayed types of
E
andF
differ,compare_strong_order_fallback(E, F)
is ill-formed.- Otherwise,
strong_order(E, F)
if it is a well-formed expression.- Otherwise,
3WAY<strong_ordering>(E, F)
([class.spaceship]) if it is a well-formed expression.- Otherwise,
compare_strong_order_fallback(E, F)
is ill-formed.The name
compare_weak_order_fallback
denotes a customization point object ([customization.point.object]). The expressioncompare_weak_order_fallback(E, F)
for some subexpressionsE
andF
is expression-equivalent ([defns.expression-equivalent]) to:
- If the decayed types of
E
andF
differ,compare_weak_order_fallback(E, F)
is ill-formed.- Otherwise,
weak_order(E, F)
if it is a well-formed expression.- Otherwise,
3WAY<weak_ordering>(E, F)
([class.spaceship]) if it is a well-formed expression.- Otherwise,
compare_weak_order_fallback(E, F)
is ill-formed.The name
compare_partial_order_fallback
denotes a customization point object ([customization.point.object]). The expressioncompare_partial_order_fallback(E, F)
for some subexpressionsE
andF
is expression-equivalent ([defns.expression-equivalent]) to:
- If the decayed types of
E
andF
differ,compare_partial_order_fallback(E, F)
is ill-formed.- Otherwise,
partial_order(E, F)
if it is a well-formed expression.- Otherwise,
3WAY<partial_ordering>(E, F)
([class.spaceship]) if it is a well-formed expression.- Otherwise,
compare_partial_order_fallback(E, F)
is ill-formed.
Change 16.13.1 [coroutine.syn]:
namespace std { [...] // 16.13.5 noop coroutine noop_coroutine_handle noop_coroutine() noexcept;
// 16.13.3.6 comparison operators: constexpr bool operator==(coroutine_handle<> x, coroutine_handle<> y) noexcept; constexpr bool operator!=(coroutine_handle<> x, coroutine_handle<> y) noexcept; constexpr bool operator<(coroutine_handle<> x, coroutine_handle<> y) noexcept; constexpr bool operator>(coroutine_handle<> x, coroutine_handle<> y) noexcept; constexpr bool operator<=(coroutine_handle<> x, coroutine_handle<> y) noexcept; constexpr bool operator>=(coroutine_handle<> x, coroutine_handle<> y) noexcept;// 16.13.6 trivial awaitables [...] }
Change 16.3.3 [coroutine.handle]:
namespace std { template <> struct coroutine_handle<void> { [...] // 16.13.3.4 resumption void operator()() const; void resume() const; void destroy() const; // comparison operators friend constexpr bool operator==(coroutine_handle x, coroutine_handle y) noexcept { return x.address() == y.address(); } friend constexpr strong_ordering operator<=>(coroutine_handle x, coroutine_handle y) noexcept { return compare_three_way()(x.address(), y.address()); } private: void* ptr; // exposition only }; [...] }
Remove 16.3.3.6 [coroutine.handle.compare] (as it's now all defined in the header):
constexpr bool operator==(coroutine_handle<> x, coroutine_handle<> y) noexcept;Returns:x.address() == y.address()
.
constexpr bool operator!=(coroutine_handle<> x, coroutine_handle<> y) noexcept;Returns:!(x == y)
.<
constexpr bool operator<(coroutine_handle<> x, coroutine_handle<> y) noexcept;Returns:less<>()(x.address(), y.address())
.
constexpr bool operator>(coroutine_handle<> x, coroutine_handle<> y) noexcept;Returns:(y < x)
.
constexpr bool operator<=(coroutine_handle<> x, coroutine_handle<> y) noexcept;Returns:!(x > y)
.
constexpr bool operator>=(coroutine_handle<> x, coroutine_handle<> y) noexcept;Returns:!(x < y)
.
No changes.
Changed operators for: error_category
, error_code
, and error_condition
Change 18.5.1 [system_error.syn]
namespace std { [...] // [syserr.errcondition.nonmembers], non-member functions error_condition make_error_condition(errc e) noexcept;
// [syserr.compare], comparison functions bool operator==(const error_code& lhs, const error_code& rhs) noexcept; bool operator==(const error_code& lhs, const error_condition& rhs) noexcept; bool operator==(const error_condition& lhs, const error_code& rhs) noexcept; bool operator==(const error_condition& lhs, const error_condition& rhs) noexcept; bool operator!=(const error_code& lhs, const error_code& rhs) noexcept; bool operator!=(const error_code& lhs, const error_condition& rhs) noexcept; bool operator!=(const error_condition& lhs, const error_code& rhs) noexcept; bool operator!=(const error_condition& lhs, const error_condition& rhs) noexcept; bool operator< (const error_code& lhs, const error_code& rhs) noexcept; bool operator< (const error_condition& lhs, const error_condition& rhs) noexcept;// [syserr.hash], hash support [...] }
Change 18.5.2.1 [syserr.errcat.overview]:
namespace std { class error_category { [...] bool operator==(const error_category& rhs) const noexcept;
bool operator!=(const error_category& rhs) const noexcept;bool operator< (const error_category& rhs) const noexcept;strong_ordering operator<=>(const error_category& rhs) const noexcept; }; [...] }
Change 18.5.2.3 [syserr.errcat.nonvirtuals]:
Returns:bool operator==(const error_category& rhs) const noexcept;
this == &rhs
.
bool operator!=(const error_category& rhs) const noexcept;Returns:!(*this == rhs)
.
bool operator<(const error_category& rhs) const noexcept;Returns:less<const error_category*>()(this, &rhs)
.
[Note:less
(19.14.7) provides a total ordering for pointers. —end note]Returns:strong_ordering operator<=>(const error_category& rhs) const noexcept;
compare_three_way()(this, &rhs)
.
[Note:compare_three_way
(cmp.object) provides a total ordering for pointers. —end note]
Change 18.5.3.1 [syserr.errcode.overview]:
namespace std { class error_code { [...] // [syserr.compare], comparison functions friend bool operator==(const error_code&, const error_code&) { see below; } friend strong_ordering operator<=>(const error_code&, const error_code&) { see below; } friend bool operator==(const error_code&, const error_condition&) { see below; } private: int val_; // exposition only const error_category* cat_; // exposition only }; [...] }
Change 18.5.4.1 [syserr.errcondition.overview]:
namespace std { class error_condition { public: [...] // [syserr.compare], comparison functions friend bool operator==(const error_condition&, const error_condition&) { see below; } friend strong_ordering operator<=>(const error_condition&, const error_condition&) { see below; } friend bool operator==(const error_condition&, const error_code&) { see below; } private: int val_; // exposition only const error_category* cat_; // exposition only }; }
Change 18.5.5 [syserr.compare]
Returns:friend bool operator==(const error_code& lhs, const error_code& rhs) noexcept;
lhs.category() == rhs.category() && lhs.value() == rhs.value()
Remarks: This function is to be found via argument-dependent lookup only.Returns:friend bool operator==(const error_code& lhs, const error_condition& rhs) noexcept;
lhs.category().equivalent(lhs.value(), rhs) || rhs.category().equivalent(lhs, rhs.value())
Remarks: This function is to be found via argument-dependent lookup only.Returns:friend bool operator==(const error_condition& lhs, const error_code& rhs) noexcept;
rhs.category().equivalent(rhs.value(), lhs) || lhs.category().equivalent(rhs, lhs.value())
Remarks: This function is to be found via argument-dependent lookup only.Returns:friend bool operator==(const error_condition& lhs, const error_condition& rhs) noexcept;
lhs.category() == rhs.category() && lhs.value() == rhs.value()
Remarks: This function is to be found via argument-dependent lookup only.
bool operator!=(const error_code& lhs, const error_code& rhs) noexcept;bool operator!=(const error_code& lhs, const error_condition& rhs) noexcept;bool operator!=(const error_condition& lhs, const error_code& rhs) noexcept;bool operator!=(const error_condition& lhs, const error_condition& rhs) noexcept;Returns:!(lhs == rhs)
.
bool operator<(const error_code& lhs, const error_code& rhs) noexcept;Returns:lhs.category() < rhs.category() || (lhs.category() == rhs.category() && lhs.value() < rhs.value())
bool operator<(const error_condition& lhs, const error_condition& rhs) noexcept;Returns:lhs.category() < rhs.category() || (lhs.category() == rhs.category() && lhs.value() < rhs.value())
Effects: Equivalent to:friend strong_ordering operator<=>(const error_code& lhs, const error_code& rhs) noexcept;
if (auto c = lhs.category() <=> rhs.category(); c != 0) return c; return lhs.value() <=> rhs.value();
Remarks: This function is to be found via argument-dependent lookup only.Effects: Equivalent to:friend strong_ordering operator<=>(const error_condition& lhs, const error_condition& rhs) noexcept;
if (auto c = lhs.category() <=> rhs.category(); c != 0) return c; return lhs.value() <=> rhs.value();
Remarks: This function is to be found via argument-dependent lookup only.
Changed operators for: pair
, tuple
, optional
, variant
, monostate
, bitset
, allocator
, unique_ptr
, shared_ptr
, memory_resource
, polymorphic_allocator
, scoped_allocator_adaptor
, function
, type_index
Change 19.2.1 [utility.syn]
#include
// see 16.10.1 namespace std { [...] // 19.4, class template pair template<class T1, class T2> struct pair; // 19.4.3, pair specialized algorithmstemplate<class T1, class T2>constexpr bool operator==(const pair<T1, T2>&, const pair<T1, T2>&);template<class T1, class T2>constexpr bool operator!=(const pair<T1, T2>&, const pair<T1, T2>&);template<class T1, class T2>constexpr bool operator< (const pair<T1, T2>&, const pair<T1, T2>&);template<class T1, class T2>constexpr bool operator> (const pair<T1, T2>&, const pair<T1, T2>&);template<class T1, class T2>constexpr bool operator<=(const pair<T1, T2>&, const pair<T1, T2>&);template<class T1, class T2>constexpr bool operator>=(const pair<T1, T2>&, const pair<T1, T2>&);[...] }
Change 19.4.2 [pairs.pair]:
namespace std { template<class T1, class T2> struct pair { [...] constexpr void swap(pair& p) noexcept(see below); friend constexpr bool operator==(const pair&, const pair&) = default; friend constexpr common_comparison_category_t<synth-3way-result<T1>, synth-3way-result<T2>> operator<=>(const pair&, const pair&) { see below } };
[...]
Requires:constexpr void swap(pair& p) noexcept(see below);
first
shall be swappable with (15.5.3.2)p.first
andsecond
shall be swappable withp.second
.
Effects: Swapsfirst
withp.first
andsecond
withp.second
.
Remarks: The expression inside noexcept is equivalent to:is_nothrow_swappable_v<first_type> && is_nothrow_swappable_v<second_type>
Effects: Equivalent to:friend constexpr common_comparison_category_t<synth-3way-result<T1>, synth-3way-result<T2>> operator<=>(const pair& lhs, const pair& rhs);
Remarks: This function is to be found via argument-dependent lookup only.if (auto c = synth-3way(lhs.first, rhs.first); c != 0) return c; return synth-3way(lhs.second, rhs.second);
Change 19.4.3 [pairs.spec].
template<class T1, class T2> constexpr bool operator==(const pair<T1, T2>& x, const pair<T1, T2>& y);Returns:x.first == y.first && x.second == y.second
.
template<class T1, class T2> constexpr bool operator!=(const pair<T1, T2>& x, const pair<T1, T2>& y);Returns:!(x == y)
.
template<class T1, class T2> constexpr bool operator<(const pair<T1, T2>& x, const pair<T1, T2>& y);Returns:x.first < y.first || (!(y.first < x.first) && x.second < y.second)
.
template<class T1, class T2> constexpr bool operator>(const pair<T1, T2>& x, const pair<T1, T2>& y);Returns:.y < x
template<class T1, class T2> constexpr bool operator<=(const pair<T1, T2>& x, const pair<T1, T2>& y);Returns:!(y < x)
.
template<class T1, class T2> constexpr bool operator>=(const pair<T1, T2>& x, const pair<T1, T2>& y);Returns:!(x < y)
.[...]template<class T1, class T2> constexpr void swap(pair<T1, T2>& x, pair<T1, T2>& y) noexcept(noexcept(x.swap(y)));
Change 19.5.2 [tuple.syn]:
namespace std { // 19.5.3, class template tuple template<class... Types> class tuple; [...] template<class T, class... Types> constexpr const T&& get(const tuple<Types...>&& t) noexcept;
// 19.5.3.8, relational operatorstemplate<class... TTypes, class... UTypes>constexpr bool operator==(const tuple<TTypes...>&, const tuple<UTypes...>&);template<class... TTypes, class... UTypes>constexpr bool operator!=(const tuple<TTypes...>&, const tuple<UTypes...>&);template<class... TTypes, class... UTypes>constexpr bool operator<(const tuple<TTypes...>&, const tuple<UTypes...>&);template<class... TTypes, class... UTypes>constexpr bool operator>(const tuple<TTypes...>&, const tuple<UTypes...>&);template<class... TTypes, class... UTypes>constexpr bool operator<=(const tuple<TTypes...>&, const tuple<UTypes...>&);template<class... TTypes, class... UTypes>constexpr bool operator>=(const tuple<TTypes...>&, const tuple<UTypes...>&);// 19.5.3.9, allocator-related traits template<class... Types, class Alloc> struct uses_allocator<tuple<Types...>, Alloc>; [...] }
Change 19.5.3 [tuple.tuple]:
namespace std { template
class tuple { public: [...] // 19.5.3.3, tuple swap constexpr void swap(tuple&) noexcept(see below ); // 19.5.3.8, tuple relational operators template<class... UTypes> friend constexpr bool operator==(const tuple&, const tuple<UTypes...>&) { see below } template<class... UTypes> friend constexpr common_comparison_category_t<synth-3way-result<Types, UTypes>...> operator<=>(const tuple&, const tuple<UTypes...>&) { see below } };
Change 19.5.3.8 [tuple.rel]:
template<class... TTypes, class... UTypes> constexpr bool operator==(const tuple<TTypes...>& t, const tuple<UTypes...>& u);Requires: For alltemplate<class... UTypes> friend constexpr bool operator==(const tuple&, const tuple<UTypes...>&)
i
, where0 <= i
andi < sizeof...(
,TTypesTypes)get<i>(t) == get<i>(u)
is a well-formed expression returning a type that is convertible tobool
.sizeof...(
.TTypesTypes) == sizeof...(UTypes)
Returns:true
ifget<i>(t) == get<i>(u)
for alli
, otherwisefalse
. For any two zero-length tuplese
andf
,e == f
returnstrue
.
Effects: The elementary comparisons are performed in order from the zeroth index upwards. No comparisons or element accesses are performed after the first equality comparison that evaluates tofalse
.
Remarks: This function is to be found via argument-dependent lookup only.
template<class... TTypes, class... UTypes>constexpr bool operator!=(const tuple<TTypes...>& t, const tuple<UTypes...>& u);Returns:!(t == u)
.
template<class... TTypes, class... UTypes>constexpr bool operator<(const tuple<TTypes...>& t, const tuple<UTypes...>& u);Requires: For alltemplate<class... UTypes> friend constexpr common_comparison_category_t<synth-3way-result<Types, UTypes>...> operator<=>(const tuple& t, const tuple<UTypes...>& u);
i
, where0 <= i
andi < sizeof...(Types)
,bothget<i>(t) < get<i>(u)
andget<i>(u) < get<i>(t)
are well-formed expressions returning types that are convertible tobool
synth-3way(get<i>(t), get<i>(u))
is a well-formed expression.sizeof...(
.TTypesTypes) == sizeof...(UTypes)
Returns: The result of a lexicographical comparison betweent
andu
. The result is defined as:(bool)(get<0>(t) < get<0>(u)) || (!(bool)(get<0>(u) < get<0>(t)) && ttail < utail)
, wherertail
for some tupler
is a tuple containing all but the first element ofr
. For any two zero-length tuplese
andf
,e < f
returnsfalse
.
Effects: Performs a lexicographical comparison betweent
andu
. For any two zero-length tuplest
andu
,t <=> u
returnsstrong_ordering::equal
. Otherwise, equivalent to:whereauto c = synth-3way(get<0>(t), get<0>(u)); return (c != 0) ? c : (ttail <=> utail);
rtail
for some tupler
is a tuple containing all but the first element ofr
.
Remarks: This function is to be found via argument-dependent lookup only.
template<class... TTypes, class... UTypes> constexpr bool operator>(const tuple<TTypes...>& t, const tuple<UTypes...>& u);Returns:.u < t
template<class... TTypes, class... UTypes> constexpr bool operator<=(const tuple<TTypes...>& t, const tuple<UTypes...>& u);Returns:!(u < t)
template<class... TTypes, class... UTypes> constexpr bool operator>=(const tuple<TTypes...>& t, const tuple<UTypes...>& u);Returns:!(t < u)
[Note: The above definitions for comparison functions do not requirettail
(orutail
) to be constructed. It may not even be possible, ast
andu
are not required to be copy constructible. Also, all comparison functions are short circuited; they do not perform element accesses beyond what is required to determine the result of the comparison. —end note]
Change 19.6.2 [optional.syn]:
namespace std { [...] // [optional.bad.access], class bad_optional_access class bad_optional_access;
// [optional.relops], relational operators template<class T, class U> constexpr bool operator==(const optional<T>&, const optional<U>&); template<class T, class U> constexpr bool operator!=(const optional<T>&, const optional<U>&); template<class T, class U> constexpr bool operator<(const optional<T>&, const optional<U>&); template<class T, class U> constexpr bool operator>(const optional<T>&, const optional<U>&); template<class T, class U> constexpr bool operator<=(const optional<T>&, const optional<U>&); template<class T, class U> constexpr bool operator>=(const optional<T>&, const optional<U>&); // [optional.nullops], comparison with nullopt template<class T> constexpr bool operator==(const optional<T>&, nullopt_t) noexcept; template<class T> constexpr bool operator==(nullopt_t, const optional<T>&) noexcept; template<class T> constexpr bool operator!=(const optional<T>&, nullopt_t) noexcept; template<class T> constexpr bool operator!=(nullopt_t, const optional<T>&) noexcept; template<class T> constexpr bool operator<(const optional<T>&, nullopt_t) noexcept; template<class T> constexpr bool operator<(nullopt_t, const optional<T>&) noexcept; template<class T> constexpr bool operator>(const optional<T>&, nullopt_t) noexcept; template<class T> constexpr bool operator>(nullopt_t, const optional<T>&) noexcept; template<class T> constexpr bool operator<=(const optional<T>&, nullopt_t) noexcept; template<class T> constexpr bool operator<=(nullopt_t, const optional<T>&) noexcept; template<class T> constexpr bool operator>=(const optional<T>&, nullopt_t) noexcept; template<class T> constexpr bool operator>=(nullopt_t, const optional<T>&) noexcept; // [optional.comp_with_t], comparison with T template<class T, class U> constexpr bool operator==(const optional<T>&, const U&); template<class T, class U> constexpr bool operator==(const T&, const optional<U>&); template<class T, class U> constexpr bool operator!=(const optional<T>&, const U&); template<class T, class U> constexpr bool operator!=(const T&, const optional<U>&); template<class T, class U> constexpr bool operator<(const optional<T>&, const U&); template<class T, class U> constexpr bool operator<(const T&, const optional<U>&); template<class T, class U> constexpr bool operator>(const optional<T>&, const U&); template<class T, class U> constexpr bool operator>(const T&, const optional<U>&); template<class T, class U> constexpr bool operator<=(const optional<T>&, const U&); template<class T, class U> constexpr bool operator<=(const T&, const optional<U>&); template<class T, class U> constexpr bool operator>=(const optional<T>&, const U&); template<class T, class U> constexpr bool operator>=(const T&, const optional<U>&);// [optional.specalg], specialized algorithms template<class T> void swap(optional<T>&, optional<T>&) noexcept(see below); [...] }
Change 19.6.3 [optional.optional]:
namespace std { template<class T> class optional { public: [...] // [optional.mod], modifiers void reset() noexcept; // [optional.relops], relational operators template<class U> friend constexpr bool operator==(const optional&, const optional<U>&) { see below } template<class U> friend constexpr bool operator!=(const optional&, const optional<U>&) { see below } template<class U> friend constexpr bool operator<(const optional&, const optional<U>&) { see below } template<class U> friend constexpr bool operator>(const optional&, const optional<U>&) { see below } template<class U> friend constexpr bool operator<=(const optional&, const optional<U>&) { see below } template<class U> friend constexpr bool operator>=(const optional&, const optional<U>&) { see below } template<ThreeWayComparableWith<T> U> friend constexpr compare_three_way_result_t<T,U> operator<=>(const optional&, const optional<U>&) { see below } // comparison with nullopt friend constexpr bool operator==(const optional& x, nullopt_t) { return !x; } friend constexpr strong_ordering operator<=>(const optional& x, nullopt_t) { return bool(x) <=> false; } // [optional.comp_with_t], comparison with T template<class U> friend constexpr bool operator==(const optional&, const U&) { see below } template<class U> friend constexpr bool operator==(const U&, const optional&) { see below } template<class U> friend constexpr bool operator!=(const optional&, const U&) { see below } template<class U> friend constexpr bool operator!=(const U&, const optional&) { see below } template<class U> friend constexpr bool operator<(const optional&, const U&) { see below } template<class U> friend constexpr bool operator<(const U&, const optional&) { see below } template<class U> friend constexpr bool operator>(const optional&, const U&) { see below } template<class U> friend constexpr bool operator>(const U&, const optional&) { see below } template<class U> friend constexpr bool operator<=(const optional&, const U&) { see below } template<class U> friend constexpr bool operator<=(const U&, const optional&) { see below } template<class U> friend constexpr bool operator>=(const optional&, const U&) { see below } template<class U> friend constexpr bool operator>=(const U&, const optional&) { see below } template<ThreeWayComparableWith<T> U> friend constexpr compare_three_way_result_t<T,U> operator<=>(const optional&, const U&) { see below } private: T *val; // exposition only }; }
Change 19.6.6 [optional.relops]:
template<
class T,class U> friend constexpr bool operator==(const optional<T>& x, const optional<U>& y);RequiresMandates: The expression*x == *y
shall be well-formed and its result shall be convertible tobool
. [Note:T
need not beCpp17EqualityComparable
. —end note]
Returns: Ifbool(x) != bool(y)
,false
; otherwise ifbool(x) == false
,true
; otherwise*x == *y
.
Remarks: Specializations of this function template for which*x == *y
is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator!=(const optional<T>& x, const optional<U>& y);RequiresMandates: The expression*x != *y
shall be well-formed and its result shall be convertible tobool
.
Returns: Ifbool(x) != bool(y)
,true
; otherwise, ifbool(x) == false
,false
; otherwise*x != *y
.
Remarks: Specializations of this function template for which*x != *y
is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator<(const optional<T>& x, const optional<U>& y);RequiresMandates:*x < *y
shall be well-formed and its result shall be convertible tobool
.
Returns: If!y
,false
; otherwise, if!x
,true
; otherwise*x < *y
.
Remarks: Specializations of this function template for which*x < *y
is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator>(const optional<T>& x, const optional<U>& y);RequiresMandates: The expression*x > *y
shall be well-formed and its result shall be convertible tobool
.
Returns: If!x
,false
; otherwise, if!y
,true
; otherwise*x > *y
. Remarks: Specializations of this function template for which*x > *y
is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator<=(const optional<T>& x, const optional<U>& y);RequiresMandates: The expression*x <= *y
shall be well-formed and its result shall be convertible tobool
.
Returns: If!x
,true
; otherwise, if!y
,false
; otherwise*x <= *y
.
Remarks: Specializations of this function template for which*x <= *y
is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.template<
class T, class U> friend constexpr bool operator>=(const optional<T>& x, const optional<U>& y);RequiresMandates: The expression*x >= *y
shall be well-formed and its result shall be convertible tobool
.
Returns: If!y
,true
; otherwise, if!x
,false
; otherwise*x >= *y
.
Remarks: Specializations of this function template for which*x >= *y
is a core constant expression shall be constexpr functions. This function is to be found via argument-dependent lookup only.Returns: Iftemplate<ThreeWayComparableWith<T> U> friend constexpr compare_three_way_result_t<T,U> operator<=>(const optional& x, const optional<U>& y);
x && y
,*x <=> *y
; otherwisebool(x) <=> bool(y)
.
Remarks: Specializations of this function template for which*x <=> *y
is a core constant expression shall beconstexpr
functions. This function is to be found via argument-dependent lookup only.
Remove 19.6.7 [optional.nullops] (it is now fully expressed by the two hidden friends defined in the header):
template<class T> constexpr bool operator==(const optional<T>& x, nullopt_t) noexcept;template<class T> constexpr bool operator==(nullopt_t, const optional<T>& x) noexcept;Returns:!x
.
template<class T> constexpr bool operator!=(const optional<T>& x, nullopt_t) noexcept; template<class T> constexpr bool operator!=(nullopt_t, const optional<T>& x) noexcept;Returns:[...]bool(x)
.
Change 19.6.8 [optional.comp_with_t]:
template<
class T,class U> friend constexpr bool operator==(const optional<T>& x, const U& v);RequiresMandates: The expression*x == v
shall be well-formed and its result shall be convertible tobool
. [Note:T
need not beCpp17EqualityComparable
. —end note]
Effects: Equivalent to:return bool(x) ? *x == v : false;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator==(constTU& v, const optional<U>& x);RequiresMandates: The expressionv == *x
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? v == *x : false;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator!=(const optional<T>& x, const U& v);RequiresMandates: The expression*x != v
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? *x != v : true;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator!=(constTU& v, const optional<U>& x);RequiresMandates: The expressionv != *x
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? v != *x : true;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator<(const optional<T>& x, const U& v);RequiresMandates: The expression*x < v
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? *x < v : true;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator<(constTU& v, const optional<U>& x);RequiresMandates: The expressionv < *x
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? v < *x : false;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator>(const optional<T>& x, const U& v);RequiresMandates: The expression*x > v
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? *x > v : false;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator>(constTU& v, const optional<U>& x);RequiresMandates: The expressionv > *x
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? v > *x : true;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator<=(const optional<T>& x, const U& v);RequiresMandates: The expression*x <= v
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? *x <= v : true;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator<=(constTU& v, const optional<U>& x);RequiresMandates: The expressionv <= *x
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? v <= *x : false;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator>=(const optional<T>& x, const U& v);RequiresMandates: The expression*x >= v
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? *x >= v : false;
Remarks: This function is to be found via argument-dependent lookup only.template<
class T,class U> friend constexpr bool operator>=(constTU& v, const optional<U>& x);RequiresMandates: The expressionv >= *x
shall be well-formed and its result shall be convertible tobool
.
Effects: Equivalent to:return bool(x) ? v >= *x : true;
Remarks: This function is to be found via argument-dependent lookup only.Effects: Equivalent to:template<ThreeWayComparableWith<T> U> friend constexpr compare_three_way_result_t<T,U> operator<=>(const optional& x, const U& v);
return bool(x) ? *x <=> v : strong_ordering::less;
Remarks: This function is to be found via argument-dependent lookup only.
Change 19.7.2 [variant.syn]:
namespace std { [...] template<class T, class... Types> constexpr add_pointer_t<T> get_if(variant<Types...>*) noexcept; template<class T, class... Types> constexpr add_pointer_t<const T> get_if(const variant<Types...>*) noexcept; // [variant.relops], relational operators
template<class... Types> constexpr bool operator==(const variant<Types...>&, const variant<Types...>&); template<class... Types> constexpr bool operator!=(const variant<Types...>&, const variant<Types...>&); template<class... Types> constexpr bool operator<(const variant<Types...>&, const variant<Types...>&); template<class... Types> constexpr bool operator>(const variant<Types...>&, const variant<Types...>&); template<class... Types> constexpr bool operator<=(const variant<Types...>&, const variant<Types...>&); template<class... Types> constexpr bool operator>=(const variant<Types...>&, const variant<Types...>&);// [variant.visit], visitation [...] // [variant.monostate], class monostate struct monostate;// [variant.monostate.relops], monostate relational operators constexpr bool operator==(monostate, monostate) noexcept; constexpr bool operator!=(monostate, monostate) noexcept; constexpr bool operator<(monostate, monostate) noexcept; constexpr bool operator>(monostate, monostate) noexcept; constexpr bool operator<=(monostate, monostate) noexcept; constexpr bool operator>=(monostate, monostate) noexcept;// [variant.specalg], specialized algorithms template<class... Types> void swap(variant<Types...>&, variant<Types...>&) noexcept(see below); [...] }
Change 19.7.3 [variant.variant]:
namespace std { template<class... Types> class variant { public: [...] // [variant.swap], swap void swap(variant&) noexcept(see below); // [variant.relops], relational operators friend constexpr bool operator==(const variant&, const variant&) { see below } friend constexpr bool operator!=(const variant&, const variant&) { see below } friend constexpr bool operator<(const variant&, const variant&) { see below } friend constexpr bool operator>(const variant&, const variant&) { see below } friend constexpr bool operator<=(const variant&, const variant&) { see below } friend constexpr bool operator>=(const variant&, const variant&) { see below } friend constexpr common_comparison_category_t<compare_three_way_result_t<Types>...> operator<=>(const variant&, const variant&) requires (ThreeWayComparable<Types> && ...) { see below } }; }
Change 19.7.6 [variant.relops]:
template<class... Types>friend constexpr bool operator==(const variant<Types...>& v, const variant<Types...>& w);RequiresMandates:get<i>(v) == get<i>(w)
is a valid expression returning a type that is convertible tobool
, for alli
.
Returns: Ifv.index() != w.index()
,false
; otherwise ifv.valueless_by_exception()
,true
; otherwiseget<i>(v) == get<i>(w)
withi
beingv.index()
.
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>friend constexpr bool operator!=(const variant<Types...>& v, const variant<Types...>& w);RequiresMandates:get<i>(v) != get<i>(w)
is a valid expression returning a type that is convertible tobool
, for alli
. Returns: Ifv.index() != w.index()
,true
; otherwise ifv.valueless_by_exception()
,false
; otherwiseget<i>(v) != get<i>(w)
withi
beingv.index()
.
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>friend constexpr bool operator<(const variant<Types...>& v, const variant<Types...>& w);RequiresMandates:get<i>(v) < get<i>(w)
is a valid expression returning a type that is convertible tobool
, for alli
.
Returns: Ifw.valueless_by_exception()
,false
; otherwise ifv.valueless_by_exception()
,true
; otherwise, ifv.index() < w.index()
,true
; otherwise ifv.index() > w.index()
,false
; otherwiseget<i>(v) < get<i>(w)
withi
beingv.index()
.
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>friend constexpr bool operator>(const variant<Types...>& v, const variant<Types...>& w);RequiresMandates:get<i>(v) > get<i>(w)
is a valid expression returning a type that is convertible tobool
, for alli
.
Returns: Ifv.valueless_by_exception()
,false
; otherwise ifw.valueless_by_exception()
,true
; otherwise, ifv.index() > w.index()
,true
; otherwise ifv.index() < w.index()
,false
; otherwiseget<i>(v) > get<i>(w)
withi
beingv.index()
.
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>friend constexpr bool operator<=(const variant<Types...>& v, const variant<Types...>& w);RequiresMandates:get<i>(v) <= get<i>(w)
is a valid expression returning a type that is convertible tobool
, for alli
.
Returns: Ifv.valueless_by_exception()
,true
; otherwise ifw.valueless_by_exception()
,false
; otherwise, ifv.index() < w.index()
,true
; otherwise ifv.index() > w.index()
,false
; otherwiseget<i>(v) <= get<i>(w)
withi
beingv.index()
.
Remarks: This function is to be found via argument-dependent lookup only.
template<class... Types>friend constexpr bool operator>=(const variant<Types...>& v, const variant<Types...>& w);RequiresMandates:get<i>(v) >= get<i>(w)
is a valid expression returning a type that is convertible tobool
, for alli
.
Returns: Ifw.valueless_by_exception()
,true
; otherwise ifv.valueless_by_exception()
,false
; otherwise, ifv.index() > w.index()
,true
; otherwise ifv.index() < w.index()
,false
; otherwiseget<i>(v) >= get<i>(w)
withi
beingv.index()
.
Remarks: This function is to be found via argument-dependent lookup only.Returns: Letconstexpr common_comparison_category_t<compare_three_way_result_t<Types>...> friend operator<=>(const variant& v, const variant& w) requires (ThreeWayComparable<Types> && ...);
c
be(v.index() + 1) <=> (w.index() + 1)
. Ifc != 0
,c
. Otherwise,get<i>(v) <=> get<i>(w)
withi
beingv.index()
.
Remarks: This function is to be found via argument-dependent lookup only.
Change 19.7.8 [variant.monostate]:
[Note: monostate objects have only a single state; they thus always compare equal. —end note]
struct monostate{};struct monostate { friend constexpr bool operator==(monostate, monostate) noexcept = default; friend constexpr strong_ordering operator<=>(monostate, monostate) noexcept = default; };
Remove 19.7.9 [variant.monostate.relops]:
constexpr bool operator==(monostate, monostate) noexcept { return true; }constexpr bool operator!=(monostate, monostate) noexcept { return false; }constexpr bool operator<(monostate, monostate) noexcept { return false; }constexpr bool operator>(monostate, monostate) noexcept { return false; }constexpr bool operator<=(monostate, monostate) noexcept { return true; }constexpr bool operator>=(monostate, monostate) noexcept { return true; }[Note: monostate objects have only a single state; they thus always compare equal. —end note]
Change 19.9.2 [template.bitset]:
namespace std { template<size_t N> class bitset { public: [...] constexpr size_t size() const noexcept; bool operator==(const bitset<N>& rhs) const noexcept;
bool operator!=(const bitset<N>& rhs) const noexcept;bool test(size_t pos) const; [...] }; }
Change 19.9.2.2 [bitset.members]:
Returns:bool operator==(const bitset<N>& rhs) const noexcept;
true
if the value of each bit in*this
equals the value of the corresponding bit inrhs
.bool operator!=(const bitset<N>& rhs) const noexcept;
Returns:true
if!(*this == rhs)
.
Change 19.10.2 [memory.syn]:
namespace std { [...] // [default.allocator], the default allocator template<class T> class allocator;
template<class T, class U> bool operator==(const allocator<T>&, const allocator<U>&) noexcept; template<class T, class U> bool operator!=(const allocator<T>&, const allocator<U>&) noexcept;[...] template<class T, class D> void swap(unique_ptr<T, D>& x, unique_ptr<T, D>& y) noexcept;template<class T1, class D1, class T2, class D2> bool operator==(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y); template<class T1, class D1, class T2, class D2> bool operator!=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y); template<class T1, class D1, class T2, class D2> bool operator<(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y); template<class T1, class D1, class T2, class D2> bool operator>(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y); template<class T1, class D1, class T2, class D2> bool operator<=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y); template<class T1, class D1, class T2, class D2> bool operator>=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y); template<class T, class D> bool operator==(const unique_ptr<T, D>& x, nullptr_t) noexcept; template<class T, class D> bool operator==(nullptr_t, const unique_ptr<T, D>& y) noexcept; template<class T, class D> bool operator!=(const unique_ptr<T, D>& x, nullptr_t) noexcept; template<class T, class D> bool operator!=(nullptr_t, const unique_ptr<T, D>& y) noexcept; template<class T, class D> bool operator<(const unique_ptr<T, D>& x, nullptr_t); template<class T, class D> bool operator<(nullptr_t, const unique_ptr<T, D>& y); template<class T, class D> bool operator>(const unique_ptr<T, D>& x, nullptr_t); template<class T, class D> bool operator>(nullptr_t, const unique_ptr<T, D>& y); template<class T, class D> bool operator<=(const unique_ptr<T, D>& x, nullptr_t); template<class T, class D> bool operator<=(nullptr_t, const unique_ptr<T, D>& y); template<class T, class D> bool operator>=(const unique_ptr<T, D>& x, nullptr_t); template<class T, class D> bool operator>=(nullptr_t, const unique_ptr<T, D>& y);template<class E, class T, class Y, class D> basic_ostream<E, T>& operator<<(basic_ostream<E, T>& os, const unique_ptr<Y, D>& p); [...]// [util.smartptr.shared.cmp], shared_ptr comparisonstemplate<class T, class U> bool operator==(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept; template<class T, class U> bool operator!=(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept; template<class T, class U> bool operator<(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept; template<class T, class U> bool operator>(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept; template<class T, class U> bool operator<=(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept; template<class T, class U> bool operator>=(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept; template<class T> bool operator==(const shared_ptr<T>& x, nullptr_t) noexcept; template<class T> bool operator==(nullptr_t, const shared_ptr<T>& y) noexcept; template<class T> bool operator!=(const shared_ptr<T>& x, nullptr_t) noexcept; template<class T> bool operator!=(nullptr_t, const shared_ptr<T>& y) noexcept; template<class T> bool operator<(const shared_ptr<T>& x, nullptr_t) noexcept; template<class T> bool operator<(nullptr_t, const shared_ptr<T>& y) noexcept; template<class T> bool operator>(const shared_ptr<T>& x, nullptr_t) noexcept; template<class T> bool operator>(nullptr_t, const shared_ptr<T>& y) noexcept; template<class T> bool operator<=(const shared_ptr<T>& x, nullptr_t) noexcept; template<class T> bool operator<=(nullptr_t, const shared_ptr<T>& y) noexcept; template<class T> bool operator>=(const shared_ptr<T>& x, nullptr_t) noexcept; template<class T> bool operator>=(nullptr_t, const shared_ptr<T>& y) noexcept;// [util.smartptr.shared.spec], shared_ptr specialized algorithms template<class T> void swap(shared_ptr<T>& a, shared_ptr<T>& b) noexcept; [...] }
Change 19.10.10 [default.allocator]:
namespace std { template<class T> class allocator { public: using value_type = T; using size_type = size_t; using difference_type = ptrdiff_t; using propagate_on_container_move_assignment = true_type; using is_always_equal = true_type; constexpr allocator() noexcept; constexpr allocator(const allocator&) noexcept; template<class U> constexpr allocator(const allocator<U>&) noexcept; ~allocator(); allocator& operator=(const allocator&) = default; [[nodiscard]] T* allocate(size_t n); void deallocate(T* p, size_t n); template<class U> friend bool operator==(const allocator&, const allocator<U>&) { return true; } }; }
Remove 19.10.10.2 [allocator.globals]:
template<class T, class U> bool operator==(const allocator<T>&, const allocator<U>&) noexcept;Returns:true
.template<class T, class U> bool operator!=(const allocator<T>&, const allocator<U>&) noexcept;
Returns:false
.
Change 19.11.1.2 [unique.ptr.single]:
namespace std { template<class T, class D = default_delete<T>> class unique_ptr { public: using pointer = see below; using element_type = T; using deleter_type = D; [...] // disable copy from lvalue unique_ptr(const unique_ptr&) = delete; unique_ptr& operator=(const unique_ptr&) = delete; // [unique.ptr.special] Specialized algorithms template<class T2, class D2> friend bool operator==(const unique_ptr& x, const unique_ptr<T2, D2>& y) { return x.get() == y.get(); } template<class T2, class D2> friend bool operator<(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below } template<class T2, class D2> friend bool operator>(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below } template<class T2, class D2> friend bool operator<=(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below } template<class T2, class D2> friend bool operator>=(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below } template<class T2, class D2> requires ThreeWayComparableWith<pointer, typename unique_ptr<T2, D2>::pointer> friend auto operator<=>(const unique_ptr& x, const unique_ptr<T2, D2>& y) { return compare_three_way()(x.get(), y.get()); } friend bool operator==(const unique_ptr& x, nullptr_t) noexcept { return !x; } friend auto operator<=>(const unique_ptr& x, nullptr_t) requires ThreeWayComparableWith<pointer, nullptr_t> { return compare_three_way()(x.get(), nullptr); } }; }
Change 19.11.1.3 [unique.ptr.runtime]:
namespace std { template
class unique_ptr { public: [...] // disable copy from lvalue unique_ptr(const unique_ptr&) = delete; unique_ptr& operator=(const unique_ptr&) = delete; // [unique.ptr.special] Specialized algorithms template<class T2, class D2> friend bool operator==(const unique_ptr& x, const unique_ptr<T2, D2>& y) { return x.get() == y.get(); } template<class T2, class D2> friend bool operator<(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below } template<class T2, class D2> friend bool operator>(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below } template<class T2, class D2> friend bool operator<=(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below } template<class T2, class D2> friend bool operator>=(const unique_ptr& x, const unique_ptr<T2, D2>& y) { see below } template<class T2, class D2> requires ThreeWayComparableWith<pointer, typename unique_ptr<T2, D2>::pointer> friend auto operator<=>(const unique_ptr& x, const unique_ptr<T2, D2>& y) { return compare_three_way()(x.get(), y.get()); } friend bool operator==(const unique_ptr& x, nullptr_t) noexcept { return !x; } friend auto operator<=>(const unique_ptr& x, nullptr_t) requires ThreeWayComparableWith<pointer, nullptr_t> { return compare_three_way()(x.get(), nullptr); } }; }
Change 19.11.1.5 [unique.ptr.special]:
Remarks: This function shall not participate in overload resolution unlesstemplate<class T, class D> void swap(unique_ptr<T, D>& x, unique_ptr<T, D>& y) noexcept;
is_swappable_v<D>
istrue
. Effects: Callsx.swap(y)
.
template<class T1, class D1, class T2, class D2>bool operator==(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);Returns:x.get() == y.get()
.
template<class T1, class D1, class T2, class D2> bool operator!=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);Returns:x.get() != y.get()
.Requires: Lettemplate<
class T1, class D1,class T2, class D2> friend bool operator<(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);CT
denotecommon_type_t<
Then the specializationtypename unique_ptr<T1, D1>::pointer, typename unique_ptr<T2, D2>::pointer>less<CT>
shall be a function object type that induces a strict weak ordering on the pointer values.
Returns:less<CT>()(x.get(), y.get())
.
Remarks: Ifunique_ptr<T1, D1>::pointer
is not implicitly convertible toCT
orunique_ptr<T2, D2>::pointer
is not implicitly convertible toCT
, the program is ill-formed.
Mandates:pointer
andunique_ptr<T2, D2>::pointer
are implicitly convertible toCT
.
Remarks: This function is to be found via argument-dependent lookup only.Returns:template<
class T1, class D1,class T2, class D2> friend bool operator>(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);y < x
.
Remarks: This function is to be found via argument-dependent lookup only.Returns:template<
class T1, class D1,class T2, class D2> friend bool operator<=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);!(y < x)
.
Remarks: This function is to be found via argument-dependent lookup only.Returns:template<
class T1, class D1,class T2, class D2> friend bool operator>=(const unique_ptr<T1, D1>& x, const unique_ptr<T2, D2>& y);!(x < y)
.
Remarks: This function is to be found via argument-dependent lookup only.
Change 19.11.3, [util.smartptr.shared]:
namespace std { template<class T> class shared_ptr { [...] // [util.smartptr.shared.obs], observers element_type* get() const noexcept; T& operator*() const noexcept; T* operator->() const noexcept; element_type& operator[](ptrdiff_t i) const; long use_count() const noexcept; explicit operator bool() const noexcept; template<class U> bool owner_before(const shared_ptr<U>& b) const noexcept; template<class U> bool owner_before(const weak_ptr<U>& b) const noexcept; // [util.smartptr.shared.cmp], shared_ptr comparisons template<class U> friend bool operator==(const shared_ptr& a, const shared_ptr<U>& b) noexcept { return a.get() == b.get(); } template<class U> friend strong_ordering operator<=>(const shared_ptr& a, const shared_ptr<U>& b) noexcept { return compare_three_way()(a.get(), b.get()); } friend bool operator==(const shared_ptr& a, nullptr_t) noexcept { return !a; } friend strong_ordering operator<=>(const shared_ptr& a, nullptr_t) noexcept { return compare_three_way()(a.get(), nullptr); } }; }
Remove all of 19.11.3.7 [util.smartptr.shared.cmp]:
template<class T, class U>bool operator==(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;Returns:a.get() == b.get()
.
template<class T, class U> bool operator<(const shared_ptr<T>& a, const shared_ptr<U>& b) noexcept;Returns:less<>()(a.get(), b.get())
.
[...]
Change 19.12.1 [mem.res.syn]:
namespace std::pmr { // [mem.res.class], class memory_resource class memory_resource;
bool operator==(const memory_resource& a, const memory_resource& b) noexcept;bool operator!=(const memory_resource& a, const memory_resource& b) noexcept;// [mem.poly.allocator.class], class template polymorphic_allocator template<class Tp> class polymorphic_allocator;template<class T1, class T2>bool operator==(const polymorphic_allocator<T1>& a,const polymorphic_allocator<T2>& b) noexcept;template<class T1, class T2>bool operator!=(const polymorphic_allocator<T1>& a,const polymorphic_allocator<T2>& b) noexcept;// [mem.res.global], global memory resources memory_resource* new_delete_resource() noexcept; [...] }
Change 19.12.2 [mem.res.class]:
namespace std::pmr { class memory_resource { static constexpr size_t max_align = alignof(max_align_t); // exposition only public: memory_resource() = default; memory_resource(const memory_resource&) = default; virtual ~memory_resource(); memory_resource& operator=(const memory_resource&) = default; [[nodiscard]] void* allocate(size_t bytes, size_t alignment = max_align); void deallocate(void* p, size_t bytes, size_t alignment = max_align); bool is_equal(const memory_resource& other) const noexcept; friend bool operator==(const memory_resource&, const memory_resource&) noexcept { see below } private: virtual void* do_allocate(size_t bytes, size_t alignment) = 0; virtual void do_deallocate(void* p, size_t bytes, size_t alignment) = 0; virtual bool do_is_equal(const memory_resource& other) const noexcept = 0; }; }
Change 19.12.2.3 [mem.res.eq]:
Returns:friend bool operator==(const memory_resource& a, const memory_resource& b) noexcept;
&a == &b || a.is_equal(b)
.
bool operator!=(const memory_resource& a, const memory_resource& b) noexcept;Returns:!(a == b)
.
Change 19.12.3 [mem.poly.allocator.class]:
namespace std::pmr { template<class Tp> class polymorphic_allocator { memory_resource* memory_rsrc; // exposition only public: [...] memory_resource* resource() const; template<class T2> friend bool operator==(const polymorphic_allocator& a, const polymorphic_allocator<T2>& b) { see below } }; }
Change 19.12.3.3 [mem.poly.allocator.eq]:
Returns:template<
class T1,class T2> friend bool operator==(const polymorphic_allocator<T1>& a, const polymorphic_allocator<T2>& b) noexcept;*a.resource() == *b.resource()
.
Remarks: This function is to be found via argument-dependent lookup only.
template<class T1, class T2>bool operator!=(const polymorphic_allocator<T1>& a,const polymorphic_allocator<T2>& b) noexcept;Returns:!(a == b)
.
Change 19.13.1 [allocator.adaptor.syn]:
namespace std { // class template scoped allocator adaptor template<class OuterAlloc, class... InnerAlloc> class scoped_allocator_adaptor;
// [scoped.adaptor.operators], scoped allocator operatorstemplate<class OuterA1, class OuterA2, class... InnerAllocs>bool operator==(const scoped_allocator_adaptor<OuterA1, InnerAllocs...>& a,const scoped_allocator_adaptor<OuterA2, InnerAllocs...>& b) noexcept;template<class OuterA1, class OuterA2, class... InnerAllocs>bool operator!=(const scoped_allocator_adaptor<OuterA1, InnerAllocs...>& a,const scoped_allocator_adaptor<OuterA2, InnerAllocs...>& b) noexcept;}
namespace std { template<class OuterAlloc, class... InnerAllocs> class scoped_allocator_adaptor : public OuterAlloc { [...] scoped_allocator_adaptor select_on_container_copy_construction() const; // [scoped.adaptor.operators], scoped allocator operators template<class Outer2> friend bool operator==(const scoped_allocator_adaptor& a, const scoped_allocator_adaptor<Outer2, InnerAllocs...>& b) noexcept { see below } }; template<class OuterAlloc, class... InnerAllocs> scoped_allocator_adaptor(OuterAlloc, InnerAllocs...) -> scoped_allocator_adaptor<OuterAlloc, InnerAllocs...>; }
Change 19.13.5 [scoped.adaptor.operators]:
Returns: Iftemplate<
class OuterA1,class OuterA2, class... InnerAllocs> friend bool operator==(const scoped_allocator_adaptor<OuterA1, InnerAllocs...>& a, const scoped_allocator_adaptor<OuterA2, InnerAllocs...>& b) noexcept;sizeof...(InnerAllocs)
is zero,
a.outer_allocator() == b.outer_allocator()
otherwise
a.outer_allocator() == b.outer_allocator() && a.inner_allocator() == b.inner_allocator()
Remarks: This function is to be found via argument-dependent lookup only.
template<class OuterA1, class OuterA2, class... InnerAllocs>bool operator!=(const scoped_allocator_adaptor<OuterA1, InnerAllocs...>& a,const scoped_allocator_adaptor<OuterA2, InnerAllocs...>& b) noexcept;Returns:!(a == b)
.
Change 19.14.1 [functional.syn]
namespace std { [...]
template<class R, class... ArgTypes>bool operator==(const function<R(ArgTypes...)>&, nullptr_t) noexcept;template<class R, class... ArgTypes>bool operator==(nullptr_t, const function<R(ArgTypes...)>&) noexcept;template<class R, class... ArgTypes>bool operator!=(const function<R(ArgTypes...)>&, nullptr_t) noexcept;template<class R, class... ArgTypes>bool operator!=(nullptr_t, const function<R(ArgTypes...)>&) noexcept;// [func.search], searchers [...] }
Change 19.14.8 [range.cmp]/2 to add <=>
:
There is an implementation-defined strict total ordering over all pointer values of a given type. This total ordering is consistent with the partial order imposed by the builtin operators
<
,>
,<=
,and>=
, and<=>
.
Change 19.14.16.2 [func.wrap.func]:
namespace std { template<class> class function; // not defined template<class R, class... ArgTypes> class function<R(ArgTypes...)> { public: using result_type = R; [...] // [func.wrap.func.targ], function target access const type_info& target_type() const noexcept; template<class T> T* target() noexcept; template<class T> const T* target() const noexcept; friend bool operator==(const function& f, nullptr_t) noexcept { return !f; } }; [...]
// [func.wrap.func.nullptr], Null pointer comparisonstemplate<class R, class... ArgTypes>bool operator==(const function<R(ArgTypes...)>&, nullptr_t) noexcept;template<class R, class... ArgTypes>bool operator==(nullptr_t, const function<R(ArgTypes...)>&) noexcept;template<class R, class... ArgTypes>bool operator!=(const function<R(ArgTypes...)>&, nullptr_t) noexcept;template<class R, class... ArgTypes>bool operator!=(nullptr_t, const function<R(ArgTypes...)>&) noexcept;[...] }
Remove 19.14.16.2.6 [func.wrap.func.nullptr]:
template<class R, class... ArgTypes>bool operator==(const function<R(ArgTypes...)>& f, nullptr_t) noexcept;template<class R, class... ArgTypes>bool operator==(nullptr_t, const function<R(ArgTypes...)>& f) noexcept;Returns:!f
.
template<class R, class... ArgTypes>bool operator!=(const function<R(ArgTypes...)>& f, nullptr_t) noexcept;template<class R, class... ArgTypes>bool operator!=(nullptr_t, const function<R(ArgTypes...)>& f) noexcept;Returns:(bool)f
.
Add a new row to 19.15.4.3 [meta.unary.prop], the "Type property predicates" table:
Template Condition Preconditions ... template<class T> struct has_strong_structural_equality;
The type T
has strong structural equality ([class.compare.default]).T
shall be a complete type,cv void
, or an array of unknown bound.
Change 19.17.2 [type.index.overview]. Note that the relational operators on type_index
are based on type_info::before
(effectively <
). type_info
could provide a three-way ordering function, but does not. Since an important motivation for the existence of type_index
is to be used as a key in an associative container, we do not want to pessimize <
- but do want to provide <=>
.
namespace std { class type_index { public: type_index(const type_info& rhs) noexcept; bool operator==(const type_index& rhs) const noexcept;
bool operator!=(const type_index& rhs) const noexcept;bool operator< (const type_index& rhs) const noexcept; bool operator> (const type_index& rhs) const noexcept; bool operator<= (const type_index& rhs) const noexcept; bool operator>= (const type_index& rhs) const noexcept; strong_ordering operator<=>(const type_index& rhs) const noexcept; size_t hash_code() const noexcept; const char* name() const noexcept; private: const type_info* target; // exposition only // Note that the use of a pointer here, rather than a reference, // means that the default copy/move constructor and assignment // operators will be provided and work as expected. }; }
Change 19.17.3 [type.index.members]:
Effects: Constructs atype_index(const type_info& rhs) noexcept;
type_index
object, the equivalent oftarget = &rhs
.Returns:bool operator==(const type_index& rhs) const noexcept;
*target == *rhs.target
.
bool operator!=(const type_index& rhs) const noexcept;Returns:*target != *rhs.target
.Returns:bool operator<(const type_index& rhs) const noexcept;
target->before(*rhs.target)
.Returns:bool operator>(const type_index& rhs) const noexcept;
rhs.target->before(*target)
.Returns:bool operator<=(const type_index& rhs) const noexcept;
!rhs.target->before(*target)
.Returns:bool operator>=(const type_index& rhs) const noexcept;
!target->before(*rhs.target)
.Effects: Equivalent tostrong_ordering operator<=>(const type_index& rhs) const noexcept;
if (*target == *rhs.target) return strong_ordering::equal; if (target->before(*rhs.target)) return strong_ordering::less; return strong_ordering::greater;
Returns:size_t hash_code() const noexcept;
target->hash_code()
. [...]
Changing the operators for basic_string
and basic_string_view
and adding extra type alises to the char_traits
specializations provided by the standard.
Change 20.2.3.1 [char.traits.specializations.char]:
namespace std { template<> struct char_traits<char> { using char_type = char; using int_type = int; using off_type = streamoff; using pos_type = streampos; using state_type = mbstate_t; using comparison_category = strong_ordering; [...] }; }
Change 20.2.3.2 [char.traits.specializations.char8_t]:
namespace std { template<> struct char_traits<char8_t> { using char_type = char8_t; using int_type = unsigned int; using off_type = streamoff; using pos_type = u8streampos; using state_type = mbstate_t; using comparison_category = strong_ordering; [...] }; }
Change 20.2.3.3 [char.traits.specializations.char16_t]:
namespace std { template<> struct char_traits<char16_t> { using char_type = char16_t; using int_type = uint_least16_t; using off_type = streamoff; using pos_type = u16streampos; using state_type = mbstate_t; using comparison_category = strong_ordering; [...] }; }
Change 20.2.3.4 [char.traits.specializations.char32_t]
namespace std { template<> struct char_traits<char32_t> { using char_type = char32_t; using int_type = uint_least32_t; using off_type = streamoff; using pos_type = u32streampos; using state_type = mbstate_t; using comparison_category = strong_ordering; [...] }; }
Change 20.2.3.5 [char.traits.specializations.wchar.t]
namespace std { template<> struct char_traits<wchar_t> { using char_type = wchar_t; using int_type = wint_t; using off_type = streamoff; using pos_type = wstreampos; using state_type = mbstate_t; using comparison_category = strong_ordering; [...] }; }
Change 20.3.1 [string.syn]:
#include <initializer_list> namespace std { [...]
template<class charT, class traits, class Allocator> bool operator==(const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator==(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator==(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs); template<class charT, class traits, class Allocator> bool operator!=(const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator!=(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator!=(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs); template<class charT, class traits, class Allocator> bool operator< (const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator< (const basic_string<charT, traits, Allocator>& lhs, const charT* rhs); template<class charT, class traits, class Allocator> bool operator< (const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator> (const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator> (const basic_string<charT, traits, Allocator>& lhs, const charT* rhs); template<class charT, class traits, class Allocator> bool operator> (const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator<=(const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator<=(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs); template<class charT, class traits, class Allocator> bool operator<=(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator>=(const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator>=(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs); template<class charT, class traits, class Allocator> bool operator>=(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs);[...] }
Change 20.3.2 [basic.string]/3. Insert wherever the editor deems appropriate:
namespace std { template<class charT, class traits = char_traits<charT>, class Allocator = allocator<charT>> class basic_string { [...] friend bool operator==(const basic_string& lhs, const basic_string& rhs) { see below } friend bool operator==(const basic_string& lhs, const charT* rhs) { see below } friend see below operator<=>(const basic_string& lhs, const basic_string& rhs) { see below } friend see below operator<=>(const basic_string& lhs, const charT* rhs) { see below } [...] }; }
Change 20.3.3.2 [string.cmp].
template<class charT, class traits, class Allocator> bool operator==(const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator==(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator==(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs);friend bool operator==(const basic_string& lhs, const basic_string& rhs); friend bool operator==(const basic_string& lhs, const charT* rhs);template<class charT, class traits, class Allocator> bool operator!=(const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator!=(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator!=(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs);
Effects: Equivalent to
return basic_string_view<charT, traits>(lhs) == basic_string_view<charT, traits>(rhs);
Remarks: This function is to be found via argument-dependent lookup only.
template<class charT, class traits, class Allocator> bool operator< (const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator< (const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator< (const basic_string<charT, traits, Allocator>& lhs, const charT* rhs); template<class charT, class traits, class Allocator> bool operator> (const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator> (const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator> (const basic_string<charT, traits, Allocator>& lhs, const charT* rhs); template<class charT, class traits, class Allocator> bool operator<=(const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator<=(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator<=(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs); template<class charT, class traits, class Allocator> bool operator>=(const basic_string<charT, traits, Allocator>& lhs, const basic_string<charT, traits, Allocator>& rhs) noexcept; template<class charT, class traits, class Allocator> bool operator>=(const charT* lhs, const basic_string<charT, traits, Allocator>& rhs); template<class charT, class traits, class Allocator> bool operator>=(const basic_string<charT, traits, Allocator>& lhs, const charT* rhs);friend see below operator<=>(const basic_string& lhs, const basic_string& rhs); friend see below operator<=>(const basic_string& lhs, const charT* rhs);
Effects:
Let. Equivalent to:op
be the operatorRemarks: This function is to be found via argument-dependent lookup only.return basic_string_view<charT, traits>(lhs)
op<=> basic_string_view<charT, traits>(rhs);
Change 20.4.1 [string.view.synop]:
namespace std { // [string.view.template], class template basic_string_view template<class charT, class traits = char_traits<charT>> class basic_string_view; // [string.view.comparison], non-member comparison functions
template<class charT, class traits>constexpr bool operator==(basic_string_view<charT, traits> x,basic_string_view<charT, traits> y) noexcept;template<class charT, class traits>constexpr bool operator!=(basic_string_view<charT, traits> x,basic_string_view<charT, traits> y) noexcept;template<class charT, class traits>constexpr bool operator< (basic_string_view<charT, traits> x,basic_string_view<charT, traits> y) noexcept;template<class charT, class traits>constexpr bool operator> (basic_string_view<charT, traits> x,basic_string_view<charT, traits> y) noexcept;template<class charT, class traits>constexpr bool operator<=(basic_string_view<charT, traits> x,basic_string_view<charT, traits> y) noexcept;template<class charT, class traits>constexpr bool operator>=(basic_string_view<charT, traits> x,basic_string_view<charT, traits> y) noexcept;// see [string.view.comparison], sufficient additional overloads of comparison functions[...] }
Change 20.4.2 [string.view.template], insert wherever the editor deems appropriate
template<class charT, class traits = char_traits<charT>> class basic_string_view { [...] friend constexpr bool operator==(basic_string_view, basic_string_view) noexcept { see below } friend constexpr see below operator<=>(basic_string_view, basic_string_view) noexcept { see below } [...] };
Remove the entirety of 20.4.3 [string.view.comparison]. The proposed two hidden friend declarations satisfy the requirements without needing extra wording. Replace it with the following:
Returns:friend constexpr bool operator==(basic_string_view lhs, basic_string_view rhs) noexcept;
lhs.compare(rhs) == 0
.
Remarks: This function is to be found via argument-dependent lookup only.Letfriend constexpr see below operator<=>(basic_string_view, basic_string_view) noexcept;
R
denote the typetraits::comparison_category
if it exists, otherwiseR
isweak_ordering
.
Returns:static_cast<R>(lhs.compare(rhs) <=> 0)
.
Remarks: This function is to be found via argument-dependent lookup only.
Change 21.3.2 [array.syn]:
#include <initializer_list> namespace std { // [array], class template array template<class T, size_t N> struct array;
template<class T, size_t N> constexpr bool operator==(const array<T, N>& x, const array<T, N>& y); template<class T, size_t N> constexpr bool operator!=(const array<T, N>& x, const array<T, N>& y); template<class T, size_t N> constexpr bool operator< (const array<T, N>& x, const array<T, N>& y); template<class T, size_t N> constexpr bool operator> (const array<T, N>& x, const array<T, N>& y); template<class T, size_t N> constexpr bool operator<=(const array<T, N>& x, const array<T, N>& y); template<class T, size_t N> constexpr bool operator>=(const array<T, N>& x, const array<T, N>& y);template<class T, size_t N> constexpr void swap(array<T, N>& x, array<T, N>& y) noexcept(noexcept(x.swap(y))); [...] }
Change 21.3.3 [deque.syn]:
#include <initializer_list> namespace std { // [deque], class template deque template<class T, class Allocator = allocator<T>> class deque;
template<class T, class Allocator> bool operator==(const deque<T, Allocator>& x, const deque<T, Allocator>& y); template<class T, class Allocator> bool operator!=(const deque<T, Allocator>& x, const deque<T, Allocator>& y); template<class T, class Allocator> bool operator< (const deque<T, Allocator>& x, const deque<T, Allocator>& y); template<class T, class Allocator> bool operator> (const deque<T, Allocator>& x, const deque<T, Allocator>& y); template<class T, class Allocator> bool operator<=(const deque<T, Allocator>& x, const deque<T, Allocator>& y); template<class T, class Allocator> bool operator>=(const deque<T, Allocator>& x, const deque<T, Allocator>& y);template<class T, class Allocator> void swap(deque<T, Allocator>& x, deque<T, Allocator>& y) noexcept(noexcept(x.swap(y))); [...] }
Change 21.3.4 [forward_list.syn]:
#include <initializer_list> namespace std { // [forwardlist], class template forwardlist template<class T, class Allocator = allocator<T>> class forward_list;
template<class T, class Allocator> bool operator==(const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y); template<class T, class Allocator> bool operator!=(const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y); template<class T, class Allocator> bool operator< (const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y); template<class T, class Allocator> bool operator> (const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y); template<class T, class Allocator> bool operator<=(const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y); template<class T, class Allocator> bool operator>=(const forward_list<T, Allocator>& x, const forward_list<T, Allocator>& y);template<class T, class Allocator> void swap(forward_list<T, Allocator>& x, forward_list<T, Allocator>& y) noexcept(noexcept(x.swap(y))); [...] }
Change 21.3.5 [list.syn]:
#include <initializer_list> namespace std { // [list], class template list template<class T, class Allocator = allocator<T>> class list;
template<class T, class Allocator> bool operator==(const list<T, Allocator>& x, const list<T, Allocator>& y); template<class T, class Allocator> bool operator!=(const list<T, Allocator>& x, const list<T, Allocator>& y); template<class T, class Allocator> bool operator< (const list<T, Allocator>& x, const list<T, Allocator>& y); template<class T, class Allocator> bool operator> (const list<T, Allocator>& x, const list<T, Allocator>& y); template<class T, class Allocator> bool operator<=(const list<T, Allocator>& x, const list<T, Allocator>& y); template<class T, class Allocator> bool operator>=(const list<T, Allocator>& x, const list<T, Allocator>& y);template<class T, class Allocator> void swap(list<T, Allocator>& x, list<T, Allocator>& y) noexcept(noexcept(x.swap(y))); [...] }
Change 21.3.6 [vector.syn]:
#include <initializer_list> namespace std { // [vector], class template vector template<class T, class Allocator = allocator<T>> class vector;
template<class T, class Allocator> bool operator==(const vector<T, Allocator>& x, const vector<T, Allocator>& y); template<class T, class Allocator> bool operator!=(const vector<T, Allocator>& x, const vector<T, Allocator>& y); template<class T, class Allocator> bool operator< (const vector<T, Allocator>& x, const vector<T, Allocator>& y); template<class T, class Allocator> bool operator> (const vector<T, Allocator>& x, const vector<T, Allocator>& y); template<class T, class Allocator> bool operator<=(const vector<T, Allocator>& x, const vector<T, Allocator>& y); template<class T, class Allocator> bool operator>=(const vector<T, Allocator>& x, const vector<T, Allocator>& y);template<class T, class Allocator> void swap(vector<T, Allocator>& x, vector<T, Allocator>& y) noexcept(noexcept(x.swap(y))); [...] }
Add to 21.2.1 [container.requirements.general], paragraph 4:
In Tables 62, 63, and 64
X
denotes a container class containing objects of typeT
,a
andb
denote values of typeX
,i
andj
denote values of type (possibly-const)X::iterator
,u
denotes an identifier,r
denotes a non-const value of typeX
, andrv
denotes a non-const rvalue of typeX
.
Add a row to Table 62 — Container requirements:
Expression Return
typeOperational
semanticsAssertion/note
pre/post-conditionComplexity i <=> j
strong_ordering
ifX::iterator
meets the random access iterator requirements, otherwisestrong_equality
constant
Add to 21.2.1 [container.requirements.general], paragraph 7:
In the expressions
wherei == j i != j i < j i <= j i >= j i > j i <=> j i - j
i
andj
denote objects of a container'siterator
type, either or both may be replaced by an object of the container'sconst_iterator
type referring to the same element with no change in semantics.
Remove 21.2.1 [container.requirements.general] table 64 - the optional container operations are now just <=>
instead of the four relational operators, and will be defined inline following the LWG guidance for flat_map
.
Table 64 lists operations that are provided for some types of containers but not others. Those containers for which the listed operations are provided shall implement the semantics described in Table 64 unless otherwise stated. If the iterators passed tolexicographical_compare
satisfy the constexpr iterator requirements ([iterator.requirements.general]) then the operations described in Table 64 are implemented by constexpr functions.
ExpressionReturn
typeOperational
semanticsAssertion/note
pre/post-conditionComplexity
a < bconvertible tobool
lexicographical_compare( a.begin(), a.end(), b.begin(), b.end())Requires:<
is defined for values ofT
.<
is a total ordering relationship.linear
a > bconvertible tobool
b < alinear
a <= bconvertible tobool
!(a > b)linear
a >= bconvertible tobool
!(a < b)linear[Note: The algorithmlexicographical_compare()
is defined in [algorithms]. —end note]
Change 21.3.7.1, paragraph 4 [array.overview]:
namespace std { template<class T, size_t N> struct array { [...] constexpr T * data() noexcept; constexpr const T * data() const noexcept; friend constexpr bool operator==(const array&, const array&) = default; friend constexpr synth-3way-result<value_type> operator<=>(const array& x, const array& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); } }; template<class T, class... U> array(T, U...) -> array<T, 1 + sizeof...(U)>; }
Change 21.3.8.1, paragraph 2 [deque.overview]:
namespace std { template<class T, class Allocator = allocator<T>> class deque { public: [...] void swap(deque&) noexcept(allocator_traits<Allocator>::is_always_equal::value); void clear() noexcept; friend bool operator==(const deque& x, const deque& y) { return ranges::equal(x, y); } friend synth-3way-result<value_type> operator<=>(const deque& x, const deque& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); } }; [...] }
Change 21.3.9.1, paragraph 3 [forwardlist.overview]
namespace std { template<class T, class Allocator = allocator<T>> class forward_list { public: [...] void sort(); template<class Compare> void sort(Compare comp); void reverse() noexcept; friend bool operator==(const forward_list& x, const forward_list& y) { return ranges::equal(x, y); } friend synth-3way-result<value_type> operator<=>(const forward_list& x, const forward_list& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); } }; [...] }
Change 21.3.10.1, paragraph 2 [list.overview]
namespace std { template<class T, class Allocator = allocator<T>> class list { public: [...] void sort(); template<class Compare> void sort(Compare comp); void reverse() noexcept; friend bool operator==(const list& x, const list& y) { return ranges::equal(x, y); } friend synth-3way-result<value_type> operator<=>(const list& x, const list& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); } }; [...] }
Change 21.3.11.1, paragraph 2 [vector.overview]
namespace std { template<class T, class Allocator = allocator<T>> class vector { public: [...] void swap(vector&) noexcept(allocator_traits<Allocator>::propagate_on_container_swap::value || allocator_traits<Allocator>::is_always_equal::value); void clear() noexcept; friend bool operator==(const vector& x, const vector& y) { return ranges::equal(x, y); } friend synth-3way-result<value_type> operator<=>(const vector& x, const vector& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); } }; [...] }
Change 21.3.12 [vector.bool]:
namespace std { template<class Allocator> class vector<bool, Allocator> { public: [...] static void swap(reference x, reference y) noexcept; void flip() noexcept; // flips all bits void clear() noexcept; friend bool operator==(const vector& x, const vector& y) { return ranges::equal(x, y); } friend strong_ordering operator<=>(const vector& x, const vector& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), compare_three_way()); } }; }
Change 21.4.2 [associative.map.syn]:
#include <initializer_list> namespace std { // [map], class template map template<class Key, class T, class Compare = less<Key>, class Allocator = allocator<pair<const Key, T>>> class map;
template<class Key, class T, class Compare, class Allocator> bool operator==(const map<Key, T, Compare, Allocator>& x, const map<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator!=(const map<Key, T, Compare, Allocator>& x, const map<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator< (const map<Key, T, Compare, Allocator>& x, const map<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator> (const map<Key, T, Compare, Allocator>& x, const map<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator<=(const map<Key, T, Compare, Allocator>& x, const map<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator>=(const map<Key, T, Compare, Allocator>& x, const map<Key, T, Compare, Allocator>& y);template<class Key, class T, class Compare, class Allocator> void swap(map<Key, T, Compare, Allocator>& x, map<Key, T, Compare, Allocator>& y) noexcept(noexcept(x.swap(y))); template <class Key, class T, class Compare, class Allocator, class Predicate> void erase_if(map<Key, T, Compare, Allocator>& c, Predicate pred); // [multimap], class template multimap template<class Key, class T, class Compare = less<Key>, class Allocator = allocator<pair<const Key, T>>> class multimap;template<class Key, class T, class Compare, class Allocator> bool operator==(const multimap<Key, T, Compare, Allocator>& x, const multimap<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator!=(const multimap<Key, T, Compare, Allocator>& x, const multimap<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator< (const multimap<Key, T, Compare, Allocator>& x, const multimap<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator> (const multimap<Key, T, Compare, Allocator>& x, const multimap<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator<=(const multimap<Key, T, Compare, Allocator>& x, const multimap<Key, T, Compare, Allocator>& y); template<class Key, class T, class Compare, class Allocator> bool operator>=(const multimap<Key, T, Compare, Allocator>& x, const multimap<Key, T, Compare, Allocator>& y);template<class Key, class T, class Compare, class Allocator> void swap(multimap<Key, T, Compare, Allocator>& x, multimap<Key, T, Compare, Allocator>& y) noexcept(noexcept(x.swap(y))); template <class Key, class T, class Compare, class Allocator, class Predicate> void erase_if(multimap<Key, T, Compare, Allocator>& c, Predicate pred); namespace pmr { template<class Key, class T, class Compare = less<Key>> using map = std::map<Key, T, Compare, polymorphic_allocator<pair<const Key, T>>>; template<class Key, class T, class Compare = less<Key>> using multimap = std::multimap<Key, T, Compare, polymorphic_allocator<pair<const Key, T>>>; } }
Change 21.4.3 [associative.set.syn]:
#include <initializer_list> namespace std { // [set], class template set template<class Key, class Compare = less<Key>, class Allocator = allocator<Key>> class set;
template<class Key, class Compare, class Allocator> bool operator==(const set<Key, Compare, Allocator>& x, const set<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator!=(const set<Key, Compare, Allocator>& x, const set<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator< (const set<Key, Compare, Allocator>& x, const set<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator> (const set<Key, Compare, Allocator>& x, const set<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator<=(const set<Key, Compare, Allocator>& x, const set<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator>=(const set<Key, Compare, Allocator>& x, const set<Key, Compare, Allocator>& y);template<class Key, class Compare, class Allocator> void swap(set<Key, Compare, Allocator>& x, set<Key, Compare, Allocator>& y) noexcept(noexcept(x.swap(y))); template <class Key, class Compare, class Allocator, class Predicate> void erase_if(set<Key, Compare, Allocator>& c, Predicate pred); // [multiset], class template multiset template<class Key, class Compare = less<Key>, class Allocator = allocator<Key>> class multiset;template<class Key, class Compare, class Allocator> bool operator==(const multiset<Key, Compare, Allocator>& x, const multiset<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator!=(const multiset<Key, Compare, Allocator>& x, const multiset<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator< (const multiset<Key, Compare, Allocator>& x, const multiset<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator> (const multiset<Key, Compare, Allocator>& x, const multiset<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator<=(const multiset<Key, Compare, Allocator>& x, const multiset<Key, Compare, Allocator>& y); template<class Key, class Compare, class Allocator> bool operator>=(const multiset<Key, Compare, Allocator>& x, const multiset<Key, Compare, Allocator>& y);template<class Key, class Compare, class Allocator> void swap(multiset<Key, Compare, Allocator>& x, multiset<Key, Compare, Allocator>& y) noexcept(noexcept(x.swap(y))); template <class Key, class Compare, class Allocator, class Predicate> void erase_if(multiset<Key, Compare, Allocator>& c, Predicate pred); namespace pmr { template<class Key, class Compare = less<Key>> using set = std::set<Key, Compare, polymorphic_allocator<Key>>; template<class Key, class Compare = less<Key>> using multiset = std::multiset<Key, Compare, polymorphic_allocator<Key>>; } }
Change 21.4.4.1 [map.overview]:
namespace std { template<class Key, class T, class Compare = less<Key>, class Allocator = allocator<pair<const Key, T>>> class map { public: [...] pair<iterator, iterator> equal_range(const key_type& x); pair<const_iterator, const_iterator> equal_range(const key_type& x) const; template<class K> pair<iterator, iterator> equal_range(const K& x); template<class K> pair<const_iterator, const_iterator> equal_range(const K& x) const; friend bool operator==(const map& x, const map& y) { return ranges::equal(x, y); } friend synth-3way-result<value_type> operator<=>(const map& x, const map& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); } }; [...] }
Change 21.4.5.1 [multimap.overview]:
namespace std { template<class Key, class T, class Compare = less<Key>, class Allocator = allocator<pair<const Key, T>>> class multimap { public: [...] pair<iterator, iterator> equal_range(const key_type& x); pair<const_iterator, const_iterator> equal_range(const key_type& x) const; template<class K> pair<iterator, iterator> equal_range(const K& x); template<class K> pair<const_iterator, const_iterator> equal_range(const K& x) const; friend bool operator==(const multimap& x, const multimap& y) { return ranges::equal(x, y); } friend synth-3way-result<value_type> operator<=>(const multimap& x, const multimap& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); } }; [...] }
Change 21.4.6.1 [set.overview]:
namespace std { template<class Key, class Compare = less<Key>, class Allocator = allocator<Key>> class set { public: [...] pair<iterator, iterator> equal_range(const key_type& x); pair<const_iterator, const_iterator> equal_range(const key_type& x) const; template<class K> pair<iterator, iterator> equal_range(const K& x); template<class K> pair<const_iterator, const_iterator> equal_range(const K& x) const; friend bool operator==(const set& x, const set& y) { return ranges::equal(x, y); } friend synth-3way-result<value_type> operator<=>(const set& x, const set& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); } }; [...] }
Change 21.4.7.1 [multiset.overview]:
namespace std { template<class Key, class Compare = less<Key>, class Allocator = allocator<Key>> class multiset { public: [...] pair<iterator, iterator> equal_range(const key_type& x); pair<const_iterator, const_iterator> equal_range(const key_type& x) const; template<class K> pair<iterator, iterator> equal_range(const K& x); template<class K> pair<const_iterator, const_iterator> equal_range(const K& x) const; friend bool operator==(const multiset& x, const multiset& y) { return ranges::equal(x, y); } friend synth-3way-result<value_type> operator<=>(const multiset& x, const multiset& y) { return lexicographical_compare_three_way(x.begin(), x.end(), y.begin(), y.end(), synth-3way); } }; [...] }
Change 21.5.2 [unord.map.syn]:
#include <initializer_list> namespace std { // [unord.map], class template unordered_map template<class Key, class T, class Hash = hash<Key>, class Pred = equal_to<Key>, class Alloc = allocator<pair<const Key, T>>> class unordered_map; // [unord.multimap], class template unordered_multimap template<class Key, class T, class Hash = hash<Key>, class Pred = equal_to<Key>, class Alloc = allocator<pair<const Key, T>>> class unordered_multimap;
template<class Key, class T, class Hash, class Pred, class Alloc> bool operator==(const unordered_map<Key, T, Hash, Pred, Alloc>& a, const unordered_map<Key, T, Hash, Pred, Alloc>& b); template<class Key, class T, class Hash, class Pred, class Alloc> bool operator!=(const unordered_map<Key, T, Hash, Pred, Alloc>& a, const unordered_map<Key, T, Hash, Pred, Alloc>& b); template<class Key, class T, class Hash, class Pred, class Alloc> bool operator==(const unordered_multimap<Key, T, Hash, Pred, Alloc>& a, const unordered_multimap<Key, T, Hash, Pred, Alloc>& b); template<class Key, class T, class Hash, class Pred, class Alloc> bool operator!=(const unordered_multimap<Key, T, Hash, Pred, Alloc>& a, const unordered_multimap<Key, T, Hash, Pred, Alloc>& b);template<class Key, class T, class Hash, class Pred, class Alloc> void swap(unordered_map<Key, T, Hash, Pred, Alloc>& x, unordered_map<Key, T, Hash, Pred, Alloc>& y) noexcept(noexcept(x.swap(y))); [...] }
Change 21.5.3 [unord.set.syn]:
#include <initializer_list> namespace std { // [unord.set], class template unordered_set template<class Key, class Hash = hash<Key>, class Pred = equal_to<Key>, class Alloc = allocator<Key>> class unordered_set; // [unord.multiset], class template unordered_multiset template<class Key, class Hash = hash<Key>, class Pred = equal_to<Key>, class Alloc = allocator<Key>> class unordered_multiset;
template<class Key, class Hash, class Pred, class Alloc> bool operator==(const unordered_set<Key, Hash, Pred, Alloc>& a, const unordered_set<Key, Hash, Pred, Alloc>& b); template<class Key, class Hash, class Pred, class Alloc> bool operator!=(const unordered_set<Key, Hash, Pred, Alloc>& a, const unordered_set<Key, Hash, Pred, Alloc>& b); template<class Key, class Hash, class Pred, class Alloc> bool operator==(const unordered_multiset<Key, Hash, Pred, Alloc>& a, const unordered_multiset<Key, Hash, Pred, Alloc>& b); template<class Key, class Hash, class Pred, class Alloc> bool operator!=(const unordered_multiset<Key, Hash, Pred, Alloc>& a, const unordered_multiset<Key, Hash, Pred, Alloc>& b);template<class Key, class Hash, class Pred, class Alloc> void swap(unordered_set<Key, Hash, Pred, Alloc>& x, unordered_set<Key, Hash, Pred, Alloc>& y) noexcept(noexcept(x.swap(y))); [...] }
Change 21.5.4.1 [unord.map.overview]:
namespace std { template<class Key, class T, class Hash = hash<Key>, class Pred = equal_to<Key>, class Allocator = allocator<pair<const Key, T>>> class unordered_map { public: [...] // hash policy float load_factor() const noexcept; float max_load_factor() const noexcept; void max_load_factor(float z); void rehash(size_type n); void reserve(size_type n); friend bool operator==(const unordered_map& x, const unordered_map& y) { return ranges::equal(x, y); } }; [...] }
Change 21.5.5.1 [unord.multimap.overview]:
namespace std { template<class Key, class T, class Hash = hash<Key>, class Pred = equal_to<Key>, class Allocator = allocator<pair<const Key, T>>> class unordered_multimap { public: [...] // hash policy float load_factor() const noexcept; float max_load_factor() const noexcept; void max_load_factor(float z); void rehash(size_type n); void reserve(size_type n); friend bool operator==(const unordered_multimap& x, const unordered_multimap& y) { return ranges::equal(x, y); } }; [...] }
Change 21.5.6.1 [unord.set.overview]:
namespace std { template<class Key, class Hash = hash<Key>, class Pred = equal_to<Key>, class Allocator = allocator<Key>> class unordered_set { public: [...] // hash policy float load_factor() const noexcept; float max_load_factor() const noexcept; void max_load_factor(float z); void rehash(size_type n); void reserve(size_type n); friend bool operator==(const unordered_set& x, const unordered_set& y) { return ranges::equal(x, y); } }; [...] }
Change 21.5.7.1 [unord.multiset.overview]:
namespace std { template<class Key, class Hash = hash<Key>, class Pred = equal_to<Key>, class Allocator = allocator<Key>> class unordered_multiset { public: [...] // hash policy float load_factor() const noexcept; float max_load_factor() const noexcept; void max_load_factor(float z); void rehash(size_type n); void reserve(size_type n); friend bool operator==(const unordered_multiset& x, const unordered_multiset& y) { return ranges::equal(x, y); } }; [...] }
Change 21.6.2 [queue.syn]:
#include <initializer_list> namespace std { template<class T, class Container = deque<T>> class queue; template<class T, class Container = vector<T>, class Compare = less<typename Container::value_type>> class priority_queue;
template<class T, class Container> bool operator==(const queue<T, Container>& x, const queue<T, Container>& y); template<class T, class Container> bool operator!=(const queue<T, Container>& x, const queue<T, Container>& y); template<class T, class Container> bool operator< (const queue<T, Container>& x, const queue<T, Container>& y); template<class T, class Container> bool operator> (const queue<T, Container>& x, const queue<T, Container>& y); template<class T, class Container> bool operator<=(const queue<T, Container>& x, const queue<T, Container>& y); template<class T, class Container> bool operator>=(const queue<T, Container>& x, const queue<T, Container>& y);template<class T, class Container> void swap(queue<T, Container>& x, queue<T, Container>& y) noexcept(noexcept(x.swap(y))); template<class T, class Container, class Compare> void swap(priority_queue<T, Container, Compare>& x, priority_queue<T, Container, Compare>& y) noexcept(noexcept(x.swap(y))); }
Change 21.6.3 [stack.syn]:
#include <initializer_list> namespace std { template<class T, class Container = deque<T>> class stack;
template<class T, class Container> bool operator==(const stack<T, Container>& x, const stack<T, Container>& y); template<class T, class Container> bool operator!=(const stack<T, Container>& x, const stack<T, Container>& y); template<class T, class Container> bool operator< (const stack<T, Container>& x, const stack<T, Container>& y); template<class T, class Container> bool operator> (const stack<T, Container>& x, const stack<T, Container>& y); template<class T, class Container> bool operator<=(const stack<T, Container>& x, const stack<T, Container>& y); template<class T, class Container> bool operator>=(const stack<T, Container>& x, const stack<T, Container>& y);template<class T, class Container> void swap(stack<T, Container>& x, stack<T, Container>& y) noexcept(noexcept(x.swap(y))); }
Change 21.6.4.1 [queue.defn]:
namespace std { template<class T, class Container = deque<T>> class queue { public: using value_type = typename Container::value_type; using reference = typename Container::reference; using const_reference = typename Container::const_reference; using size_type = typename Container::size_type; using container_type = Container; protected: Container c; public: queue() : queue(Container()) {} explicit queue(const Container&); explicit queue(Container&&); template<class Alloc> explicit queue(const Alloc&); template<class Alloc> queue(const Container&, const Alloc&); template<class Alloc> queue(Container&&, const Alloc&); template<class Alloc> queue(const queue&, const Alloc&); template<class Alloc> queue(queue&&, const Alloc&); [[nodiscard]] bool empty() const { return c.empty(); } size_type size() const { return c.size(); } reference front() { return c.front(); } const_reference front() const { return c.front(); } reference back() { return c.back(); } const_reference back() const { return c.back(); } void push(const value_type& x) { c.push_back(x); } void push(value_type&& x) { c.push_back(std::move(x)); } template<class... Args> decltype(auto) emplace(Args&&... args) { return c.emplace_back(std::forward<Args>(args)...); } void pop() { c.pop_front(); } void swap(queue& q) noexcept(is_nothrow_swappable_v<Container>) { using std::swap; swap(c, q.c); } friend bool operator==(const queue& x, const queue& y) { return x.c == y.c; } friend bool operator!=(const queue& x, const queue& y) { return x.c != y.c; } friend bool operator< (const queue& x, const queue& y) { return x.c < y.c; } friend bool operator> (const queue& x, const queue& y) { return x.c > y.c; } friend bool operator<=(const queue& x, const queue& y) { return x.c <= y.c; } friend bool operator>=(const queue& x, const queue& y) { return x.c >= y.c; } friend auto operator<=>(const queue& x, const queue& y) requires ThreeWayComparable<Container> { return x.c <=> y.c; } }; [...] }
Remove 21.6.4.4 [queue.ops] (as we've now defined them all inline in the header):
template<class T, class Container> bool operator==(const queue<T, Container>& x, const queue<T, Container>& y);Returns:x.c == y.c
.
[...]
Change 21.6.6.1 [stack.defn]:
namespace std { template<class T, class Container = deque<T>> class stack { public: using value_type = typename Container::value_type; using reference = typename Container::reference; using const_reference = typename Container::const_reference; using size_type = typename Container::size_type; using container_type = Container; protected: Container c; public: stack() : stack(Container()) {} explicit stack(const Container&); explicit stack(Container&&); template<class Alloc> explicit stack(const Alloc&); template<class Alloc> stack(const Container&, const Alloc&); template<class Alloc> stack(Container&&, const Alloc&); template<class Alloc> stack(const stack&, const Alloc&); template<class Alloc> stack(stack&&, const Alloc&); [[nodiscard]] bool empty() const { return c.empty(); } size_type size() const { return c.size(); } reference top() { return c.back(); } const_reference top() const { return c.back(); } void push(const value_type& x) { c.push_back(x); } void push(value_type&& x) { c.push_back(std::move(x)); } template<class... Args> decltype(auto) emplace(Args&&... args) { return c.emplace_back(std::forward<Args>(args)...); } void pop() { c.pop_back(); } void swap(stack& s) noexcept(is_nothrow_swappable_v<Container>) { using std::swap; swap(c, s.c); } friend bool operator==(const stack& x, const stack& y) { return x.c == y.c; } friend bool operator!=(const stack& x, const stack& y) { return x.c != y.c; } friend bool operator< (const stack& x, const stack& y) { return x.c < y.c; } friend bool operator> (const stack& x, const stack& y) { return x.c > y.c; } friend bool operator<=(const stack& x, const stack& y) { return x.c <= y.c; } friend bool operator>=(const stack& x, const stack& y) { return x.c >= y.c; } friend auto operator<=>(const stack& x, const stack& y) requires ThreeWayComparable<Container> { return x.c <=> y.c; } }; [...] }
Remove 21.6.6.4 [stack.ops] (as we've now defined them all inline in the header):
template<class T, class Container> bool operator==(const stack<T, Container>& x, const stack<T, Container>& y);Returns:x.c == y.c
.
[...]
Change 22.2 [iterator.synopsis]:
#include <concepts> namespace std { [...] // [predef.iterators], predefined iterators and sentinels // [reverse.iterators], reverse iterators template<class Iterator> class reverse_iterator;
template<class Iterator1, class Iterator2> constexpr bool operator==( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator!=( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator<( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator>( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator<=( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator>=( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y);[...] // [move.iterators], move iterators and sentinels template<class Iterator> class move_iterator;template<class Iterator1, class Iterator2> constexpr bool operator==( const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator!=( const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator<( const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator>( const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator<=( const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y); template<class Iterator1, class Iterator2> constexpr bool operator>=( const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);[...] // [stream.iterators], stream iterators template<class T, class charT = char, class traits = char_traits<charT>, class Distance = ptrdiff_t> class istream_iterator;template<class T, class charT, class traits, class Distance> bool operator==(const istream_iterator<T,charT,traits,Distance>& x, const istream_iterator<T,charT,traits,Distance>& y); template<class T, class charT, class traits, class Distance> bool operator!=(const istream_iterator<T,charT,traits,Distance>& x, const istream_iterator<T,charT,traits,Distance>& y);template<class T, class charT = char, class traits = char_traits<charT>> class ostream_iterator; template<class charT, class traits = char_traits<charT>> class istreambuf_iterator;template<class charT, class traits> bool operator==(const istreambuf_iterator<charT,traits>& a, const istreambuf_iterator<charT,traits>& b); template<class charT, class traits> bool operator!=(const istreambuf_iterator<charT,traits>& a, const istreambuf_iterator<charT,traits>& b);template<class charT, class traits = char_traits<charT>> class ostreambuf_iterator; [...] }
Change 22.5.1.1 [reverse.iterator]:
namespace std { template<class Iterator> class reverse_iterator { public: [...] template<IndirectlySwappable<Iterator> Iterator2> friend constexpr void iter_swap(const reverse_iterator& x, const reverse_iterator<Iterator2>& y) noexcept(see below); // [reverse.iter.cmp] Comparisons template<class Iterator2> friend constexpr bool operator==(const reverse_iterator& x, const reverse_iterator<Iterator2>& y) { see below } template<class Iterator2> friend constexpr bool operator!=(const reverse_iterator& x, const reverse_iterator<Iterator2>& y) { see below } template<class Iterator2> friend constexpr bool operator< (const reverse_iterator& x, const reverse_iterator<Iterator2>& y) { see below } template<class Iterator2> friend constexpr bool operator> (const reverse_iterator& x, const reverse_iterator<Iterator2>& y) { see below } template<class Iterator2> friend constexpr bool operator<=(const reverse_iterator& x, const reverse_iterator<Iterator2>& y) { see below } template<class Iterator2> friend constexpr bool operator>=(const reverse_iterator& x, const reverse_iterator<Iterator2>& y) { see below } template<ThreeWayComparableWith<Iterator> Iterator2> friend constexpr compare_three_way_result_t<Iterator, Iterator2> operator<=>(const reverse_iterator& x, const reverse_iterator<Iterator2>& y) { see below } protected: Iterator current; }; }
Change 22.5.1.7 [reverse.iter.cmp]:
Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator==( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y);x.current == y.current
is well-formed and convertible tobool
.
Returns:x.current == y.current
.
Remarks: This function is to be found via argument-dependent lookup only.Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator!=( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y);x.current != y.current
is well-formed and convertible tobool
.
Returns:x.current != y.current
.
Remarks: This function is to be found via argument-dependent lookup only.Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator<( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y);x.current > y.current
is well-formed and convertible tobool
.
Returns:x.current > y.current
.
Remarks: This function is to be found via argument-dependent lookup only.Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator>( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y);x.current < y.current
is well-formed and convertible tobool
.
Returns:x.current < y.current
.
Remarks: This function is to be found via argument-dependent lookup only.Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator<=( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y);x.current >= y.current
is well-formed and convertible tobool
.
Returns:x.current >= y.current
.
Remarks: This function is to be found via argument-dependent lookup only.Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator>=( const reverse_iterator<Iterator1>& x, const reverse_iterator<Iterator2>& y);x.current <= y.current
is well-formed and convertible tobool
.
Returns:x.current <= y.current
.
Remarks: This function is to be found via argument-dependent lookup only.Returns:template<ThreeWayComparableWith<Iterator> Iterator2> friend constexpr compare_three_way_result_t<Iterator, Iterator2> operator<=>( const reverse_iterator& x, const reverse_iterator<Iterator2>& y);
y.current <=> x.current
.
Remarks: This function is more constrained than ([temp.constr.order]) each of the other relational operator function templates. This function is to be found via argument-dependent lookup only.
Change 22.5.3.1 [move.iterator]:
namespace std { template<class Iterator> class move_iterator { [...] // [move.iter.op.comp] Comparisons template<class Iterator2> friend constexpr bool operator==(const move_iterator& x, const move_iterator<Iterator2>& y) { see below } template<class Iterator2> friend constexpr bool operator<(const move_iterator& x, const move_iterator<Iterator2>& y) { see below } template<class Iterator2> friend constexpr bool operator>(const move_iterator& x, const move_iterator<Iterator2>& y) { see below } template<class Iterator2> friend constexpr bool operator<=(const move_iterator& x, const move_iterator<Iterator2>& y) { see below } template<class Iterator2> friend constexpr bool operator>=(const move_iterator& x, const move_iterator<Iterator2>& y) { see below } template<ThreeWayComparableWith<Iterator> Iterator2> friend constexpr compare_three_way_result_t<Iterator, Iterator2> operator<=>(const move_iterator& x, const move_iterator<Iterator2>& y) { see below } template<Sentinel<Iterator> S> friend constexpr bool operator==(const move_iterator& x, const move_sentinel<S>& y);
template<Sentinel<Iterator> S> friend constexpr bool operator==(const move_sentinel<S>& x, const move_iterator& y); template<Sentinel<Iterator> S> friend constexpr bool operator!=(const move_iterator& x, const move_sentinel<S>& y); template<Sentinel<Iterator> S> friend constexpr bool operator!=(const move_sentinel<S>& x, const move_iterator& y);template<SizedSentinel<Iterator> S> friend constexpr iter_difference_t<Iterator> operator-(const move_sentinel<S>& x, const move_iterator& y); [...] private: Iterator current; // exposition only }; }
Change 22.5.3.7 [move.iter.op.comp]:
Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator==(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y); template<Sentinel<Iterator> S> friend constexpr bool operator==(const move_iterator& x, const move_sentinel<S>& y); template<Sentinel<Iterator> S> friend constexpr bool operator==(const move_sentinel<S>& x, const move_iterator& y);x.base() == y.base()
is well-formed and convertible tobool
.
Returns:x.base() == y.base()
.
Remarks: This function is to be found via argument-dependent lookup only.
template<class Iterator1, class Iterator2> constexpr bool operator!=(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y); template<Sentinel<Iterator> S> friend constexpr bool operator!=(const move_iterator& x, const move_sentinel<S>& y); template<Sentinel<Iterator> S> friend constexpr bool operator!=(const move_sentinel<S>& x, const move_iterator& y);Constraints:x.base() == y.base()
is well-formed and convertible tobool
.
Returns:!(x == y)
.Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator<(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);x.base() < y.base()
is well-formed and convertible tobool
.
Returns:x.base() < y.base()
.
Remarks: This function is to be found via argument-dependent lookup only.Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator>(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);y.base() < x.base()
is well-formed and convertible tobool
.
Returns:y < x
.
Remarks: This function is to be found via argument-dependent lookup only.Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator<=(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);y.base() < x.base()
is well-formed and convertible tobool
.
Returns:!(y < x)
.
Remarks: This function is to be found via argument-dependent lookup only.Constraints:template<
class Iterator1,class Iterator2> friend constexpr bool operator>=(const move_iterator<Iterator1>& x, const move_iterator<Iterator2>& y);x.base() < y.base()
is well-formed and convertible tobool
.
Returns:!(x < y)
.
Remarks: This function is to be found via argument-dependent lookup only.Returns:template<ThreeWayComparableWith<Iterator> Iterator2> friend constexpr compare_three_way_result_t<Iterator, Iterator2> operator<=>( const move_iterator& x, const move_iterator<Iterator2>& y);
x.base() <=> y.base()
.
Remarks: This function is more constrained than ([temp.constr.order]) each of the other relational operator function templates. This function is to be found via argument-dependent lookup only.
Remove the operator!=
from 22.5.4.1 [common.iterator]:
namespace std { template<Iterator I, Sentinel<I> S> requires (!Same<I, S>) class common_iterator { public: [...] template<class I2, Sentinel<I> S2> requires Sentinel<S, I2> friend bool operator==( const common_iterator& x, const common_iterator<I2, S2>& y); template<class I2, Sentinel<I> S2> requires Sentinel<S, I2> && EqualityComparableWith<I, I2> friend bool operator==( const common_iterator& x, const common_iterator<I2, S2>& y);
template<class I2, Sentinel<I> S2> requires Sentinel<S, I2> friend bool operator!=( const common_iterator& x, const common_iterator<I2, S2>& y);[...] private: variant<I, S> v_; // exposition only }; [...] }
Remove the operator!=
from 22.5.4.6 [common.iter.cmp]:
template<class I2, Sentinel<I> S2> requires Sentinel<S, I2> friend bool operator!=( const common_iterator& x, const common_iterator<I2, S2>& y);Effects: Equivalent to:return !(x == y);
Change 22.5.6.1 [counted.iterator]:
namespace std { template<Iterator I> class counted_iterator { public: using iterator_type = I; [...] template<Common<I> I2> friend constexpr bool operator==( const counted_iterator& x, const counted_iterator<I2>& y); friend constexpr bool operator==( const counted_iterator& x, default_sentinel_t);
friend constexpr bool operator==(default_sentinel_t, const counted_iterator& x);template<Common<I> I2>friend constexpr bool operator!=(const counted_iterator& x, const counted_iterator<I2>& y);friend constexpr bool operator!=(const counted_iterator& x, default_sentinel_t y);friend constexpr bool operator!=(default_sentinel_t x, const counted_iterator& y);template<Common<I> I2> friend constexpr bool operator<( const counted_iterator& x, const counted_iterator<I2>& y); template<Common<I> I2> friend constexpr bool operator>( const counted_iterator& x, const counted_iterator<I2>& y); template<Common<I> I2> friend constexpr bool operator<=( const counted_iterator& x, const counted_iterator<I2>& y); template<Common<I> I2> friend constexpr bool operator>=( const counted_iterator& x, const counted_iterator<I2>& y); template<Common<I> I2> requires ThreeWayComparableWith<I, I2> friend constexpr compare_three_way_result_t<I, I2> operator<=>( const counted_iterator& x, const counted_iterator<I2>& y); friend constexpr iter_rvalue_reference_t<I> iter_move(const counted_iterator& i) noexcept(noexcept(ranges::iter_move(i.current))) requires InputIterator<I>; template<IndirectlySwappable<I> I2> friend constexpr void iter_swap(const counted_iterator& x, const counted_iterator<I2>& y) noexcept(noexcept(ranges::iter_swap(x.current, y.current))); private: I current = I(); // exposition only iter_difference_t<I> length = 0; // exposition only }; [...] }
Change 22.5.6.6 [counted.iter.cmp]:
Expects:template<Common<I> I2> friend constexpr bool operator==( const counted_iterator& x, const counted_iterator<I2>& y);
x
andy
refer to elements of the same sequence ([counted.iterator]).
Effects: Equivalent to: returnx.length == y.length;
Effects: Equivalent to:friend constexpr bool operator==( const counted_iterator& x, default_sentinel_t);
friend constexpr bool operator==(default_sentinel_t, const counted_iterator& x);return x.length == 0;
template<Common<I> I2> friend constexpr bool operator!=( const counted_iterator& x, const counted_iterator<I2>& y); friend constexpr bool operator!=( const counted_iterator& x, default_sentinel_t y); friend constexpr bool operator!=( default_sentinel_t x, const counted_iterator& y);Effects: Equivalent to:return !(x == y);
Expects:template<Common<I> I2> friend constexpr bool operator<( const counted_iterator& x, const counted_iterator<I2>& y);
x
andy
refer to elements of the same sequence ([counted.iterator]).
Effects: Equivalent to:return y.length < x.length;
[Note: The argument order in the Effects: element is reversed because length counts down, not up. —end note]Effects: Equivalent to:template<Common<I> I2> friend constexpr bool operator>( const counted_iterator& x, const counted_iterator<I2>& y);
return y < x;
Effects: Equivalent to:template<Common<I> I2> friend constexpr bool operator<=( const counted_iterator& x, const counted_iterator<I2>& y);
return !(y < x);
Effects: Equivalent to:template<Common<I> I2> friend constexpr bool operator>=( const counted_iterator& x, const counted_iterator<I2>& y);
return !(x < y);
Effects: Equivalent to:template<Common<I> I2> requires ThreeWayComparableWith<I, I2> friend constexpr compare_three_way_result_t<I, I2> operator<=>( const counted_iterator& x, const counted_iterator<I2>& y);
return y <=> x;
Change 22.5.7.1 [unreachable.sentinel]:
namespace std { struct unreachable_sentinel_t { template<WeaklyIncrementable I> friend constexpr bool operator==(unreachable_sentinel_t, const I&) noexcept
;{ return true; }template<WeaklyIncrementable I> friend constexpr bool operator==(const I&, unreachable_sentinel_t) noexcept; template<WeaklyIncrementable I> friend constexpr bool operator!=(unreachable_sentinel_t, const I&) noexcept; template<WeaklyIncrementable I> friend constexpr bool operator!=(const I&, unreachable_sentinel_t) noexcept;}; }
Remove 22.5.7.2 [unreachable.sentinel.cmp] (as it's now entirely defined in the synopsis):
template<WeaklyIncrementable I> friend constexpr bool operator==(unreachable_sentinel_t, const I&) noexcept; template<WeaklyIncrementable I> friend constexpr bool operator==(const I&, unreachable_sentinel_t) noexcept;Returns:false
.
template<WeaklyIncrementable I> friend constexpr bool operator!=(unreachable_sentinel_t, const I&) noexcept; template<WeaklyIncrementable I> friend constexpr bool operator!=(const I&, unreachable_sentinel_t) noexcept;Returns:true
.
Change 22.6.1 [istream.iterator]:
namespace std { template<class T, class charT = char, class traits = char_traits<charT>, class Distance = ptrdiff_t> class istream_iterator { public: using iterator_category = input_iterator_tag; using value_type = T; using difference_type = Distance; using pointer = const T*; using reference = const T&; using char_type = charT; using traits_type = traits; using istream_type = basic_istream<charT,traits>; constexpr istream_iterator(); constexpr istream_iterator(default_sentinel_t); istream_iterator(istream_type& s); istream_iterator(const istream_iterator& x) = default; ~istream_iterator() = default; istream_iterator& operator=(const istream_iterator&) = default; const T& operator*() const; const T* operator->() const; istream_iterator& operator++(); istream_iterator operator++(int); friend bool operator==(const istream_iterator& i, default_sentinel_t)
;{ return !i.in_stream; } friend bool operator==(const istream_iterator& x, const istream_iterator& y) { return x.in_stream == y.in_stream; }friend bool operator==(default_sentinel_t, const istream_iterator& i);friend bool operator!=(const istream_iterator& x, default_sentinel_t y);friend bool operator!=(default_sentinel_t x, const istream_iterator& y);private: basic_istream<charT,traits>* in_stream; // exposition only T value; // exposition only }; }
Remove the specifications for operator==
and operator!=
from 22.6.1.2 [istream.iterator.ops]:
template<class T, class charT, class traits, class Distance> bool operator==(const istream_iterator<T,charT,traits,Distance>& x, const istream_iterator<T,charT,traits,Distance>& y);Returns:x.in_stream == y.in_stream
.
friend bool operator==(default_sentinel_t, const istream_iterator& i); friend bool operator==(const istream_iterator& i, default_sentinel_t);Returns:!i.in_stream
.
template<class T, class charT, class traits, class Distance> bool operator!=(const istream_iterator<T,charT,traits,Distance>& x, const istream_iterator<T,charT,traits,Distance>& y); friend bool operator!=(default_sentinel_t x, const istream_iterator& y); friend bool operator!=(const istream_iterator& x, default_sentinel_t y);Returns:!(x == y)
Change 22.6.3 [istreambuf.iterator]:
namespace std { template<class charT, class traits = char_traits<charT>> class istreambuf_iterator { public: [...] bool equal(const istreambuf_iterator& b) const;
friend bool operator==(default_sentinel_t s, const istreambuf_iterator& i);friend bool operator==(const istreambuf_iterator& i, default_sentinel_t s);{ return i.equal(s); }friend bool operator!=(default_sentinel_t a, const istreambuf_iterator& b);friend bool operator!=(const istreambuf_iterator& a, default_sentinel_t b);friend bool operator==(const istreambuf_iterator& a, const istreambuf_iterator& b) { return a.equal(b); } private: streambuf_type* sbuf_; // exposition only }; }
Remove the specifications for operator==
and operator!=
from 22.6.3.3 [istreambuf.iterator.ops]:
template<class charT, class traits> bool operator==(const istreambuf_iterator<charT,traits>& a, const istreambuf_iterator<charT,traits>& b);Returns:a.equal(b)
.
friend bool operator==(default_sentinel_t s, const istreambuf_iterator& i); friend bool operator==(const istreambuf_iterator& i, default_sentinel_t s);Returns:i.equal(s)
.
template<class charT, class traits> bool operator!=(const istreambuf_iterator<charT,traits>& a, const istreambuf_iterator<charT,traits>& b); friend bool operator!=(default_sentinel_t a, const istreambuf_iterator& b); friend bool operator!=(const istreambuf_iterator& a, default_sentinel_t b);Returns:!a.equal(b)
.
Change 23.6.3.3 [range.iota.iterator]:
namespace std::ranges { template<class W, class Bound> struct iota_view<W, Bound>::iterator { private: W value_ = W(); // exposition only public: [...] friend constexpr bool operator==(const iterator& x, const iterator& y) requires EqualityComparable<W>;
friend constexpr bool operator!=(const iterator& x, const iterator& y)requires EqualityComparable<W>;friend constexpr bool operator<(const iterator& x, const iterator& y) requires StrictTotallyOrdered<W>; friend constexpr bool operator>(const iterator& x, const iterator& y) requires StrictTotallyOrdered<W>; friend constexpr bool operator<=(const iterator& x, const iterator& y) requires StrictTotallyOrdered<W>; friend constexpr bool operator>=(const iterator& x, const iterator& y) requires StrictTotallyOrdered<W>; friend constexpr compare_three_way_result_t<W> operator<=>(const iterator& x, const iterator& y) requires StrictTotallyOrdered<W> && ThreeWayComparable<W> [...] }; }
Change 23.6.3.3 [range.iota.iterator], paragraphs 14-20:
Effects: Equivalent to:friend constexpr bool operator==(const iterator& x, const iterator& y) requires EqualityComparable<W>;
return x.value_ == y.value_;
friend constexpr bool operator!=(const iterator& x, const iterator& y) requires EqualityComparable<W>;Effects: Equivalent to:return !(x == y);
Effects: Equivalent to:friend constexpr bool operator<(const iterator& x, const iterator& y) requires StrictTotallyOrdered<W>;
return x.value_ < y.value_;
[...]Effects: Equivalent to:friend constexpr bool operator>=(const iterator& x, const iterator& y) requires StrictTotallyOrdered<W>;
return !(x < y);
Effects: Equivalent to:friend constexpr compare_three_way_result_t<W> operator<=>(const iterator& x, const iterator& y) requires StrictTotallyOrdered<W> && ThreeWayComparable<W>;
return x.value_ <=> y.value_;
Remove obsolete equality operators 23.6.3.4 [range.iota.sentinel]:
namespace std::ranges { template<class W, class Bound> struct iota_view<W, Bound>::sentinel { private: Bound bound_ = Bound(); // exposition only public: sentinel() = default; constexpr explicit sentinel(Bound bound); friend constexpr bool operator==(const iterator& x, const sentinel& y);
friend constexpr bool operator==(const sentinel& x, const iterator& y);friend constexpr bool operator!=(const iterator& x, const sentinel& y);friend constexpr bool operator!=(const sentinel& x, const iterator& y);}; }Effects: Initializesconstexpr explicit sentinel(Bound bound);
bound_
withbound
.Effects: Equivalent to:friend constexpr bool operator==(const iterator& x, const sentinel& y);
return x.value_ == y.bound_;
friend constexpr bool operator==(const sentinel& x, const iterator& y);Effects: Equivalent to: return y == x;
friend constexpr bool operator!=(const iterator& x, const sentinel& y);Effects: Equivalent to: return !(x == y);
friend constexpr bool operator!=(const sentinel& x, const iterator& y);Effects: Equivalent to: return !(y == x);
Remove operator!=
from 23.7.4.3 [range.filter.iterator]:
namespace std::ranges { template<class V, class Pred> class filter_view<V, Pred>::iterator { private: iterator_t<V> current_ = iterator_t<V>(); // exposition only filter_view* parent_ = nullptr; // exposition only public: [...] friend constexpr bool operator==(const iterator& x, const iterator& y) requires EqualityComparable<iterator_t<V>>;
friend constexpr bool operator!=(const iterator& x, const iterator& y)requires EqualityComparable<iterator_t<V>>;[...] }; }
and
Effects: Equivalent to:friend constexpr bool operator==(const iterator& x, const iterator& y) requires EqualityComparable<iterator_t<V>>;
return x.current_ == y.current_;
friend constexpr bool operator!=(const iterator& x, const iterator& y) requires EqualityComparable<iterator_t<V>>;Effects: Equivalent to:return !(x == y);
Remove obsolete equality operators from 23.7.4.4 [range.filter.sentinel]:
namespace std::ranges { template<class V, class Pred> class filter_view<V, Pred>::sentinel { private: sentinel_t<V> end_ = sentinel_t<V>(); // exposition only public: sentinel() = default; constexpr explicit sentinel(filter_view& parent); constexpr sentinel_t<V> base() const; friend constexpr bool operator==(const iterator& x, const sentinel& y);
friend constexpr bool operator==(const sentinel& x, const iterator& y);friend constexpr bool operator!=(const iterator& x, const sentinel& y);friend constexpr bool operator!=(const sentinel& x, const iterator& y);}; }Effects: Initializesconstexpr explicit sentinel(filter_view& parent);
end_
withranges::end(parent)
.Effects: Equivalent to:constexpr sentinel_t<V> base() const;
return end_;
Effects: Equivalent to:friend constexpr bool operator==(const iterator& x, const sentinel& y);
return x.current_ == y.end_;
friend constexpr bool operator==(const sentinel& x, const iterator& y);Effects: Equivalent to:return y == x;
friend constexpr bool operator!=(const iterator& x, const sentinel& y);Effects: Equivalent to:return !(x == y);
friend constexpr bool operator!=(const sentinel& x, const iterator& y);Effects: Equivalent to:return !(y == x);
Change 23.7.5.3 [range.transform.iterator]:
namespace std::ranges { template<class V, class F> template<bool Const> class transform_view<V, F>::iterator { [...] friend constexpr bool operator==(const iterator& x, const iterator& y) requires EqualityComparable<iterator_t<Base>>;
friend constexpr bool operator!=(const iterator& x, const iterator& y)requires EqualityComparable<iterator_t<Base>>;friend constexpr bool operator<(const iterator& x, const iterator& y) requires RandomAccessRange<Base>; friend constexpr bool operator>(const iterator& x, const iterator& y) requires RandomAccessRange<Base>; friend constexpr bool operator<=(const iterator& x, const iterator& y) requires RandomAccessRange<Base>; friend constexpr bool operator>=(const iterator& x, const iterator& y) requires RandomAccessRange<Base>; friend constexpr compare_three_way_result_t<iterator_t<Base>> operator<=>(const iterator& x, const iterator& y) requires RandomAccessRange<Base>; && ThreeWayComparable<iterator_t<Base>> [...] }; }
Change 23.7.5.3 [range.transform.iterator], paragraphs 13-18:
Effects: Equivalent to:friend constexpr bool operator==(const iterator& x, const iterator& y) requires EqualityComparable<iterator_t<Base>>;
return x.current_ == y.current_;
friend constexpr bool operator!=(const iterator& x, const iterator& y) requires EqualityComparable<iterator_t<Base>>;Effects: Equivalent to:return !(x == y);
Effects: Equivalent to:friend constexpr bool operator<(const iterator& x, const iterator& y) requires RandomAccessRange<Base>;
return x.current_ < y.current_;
[...]
Effects: Equivalent to:friend constexpr bool operator>=(const iterator& x, const iterator& y) requires RandomAccessRange<Base>;
return !(x < y);
Effects: Equivalent to:friend constexpr compare_three_way_result_t<iterator_t<Base>> operator<=>(const iterator& x, const iterator& y) requires RandomAccessRange<Base>; && ThreeWayComparable<iterator_t<Base>>
return x.current_ <=> y.current_;
Remove obsolete equality operators from 23.7.5.4 [range.transform.sentinel]:
namespace std::ranges { template<class V, class F> template<bool Const> class transform_view<V, F>::sentinel { [...] friend constexpr bool operator==(const iterator<Const>& x, const sentinel& y);
friend constexpr bool operator==(const sentinel& x, const iterator<Const>& y);friend constexpr bool operator!=(const iterator<Const>& x, const sentinel& y);friend constexpr bool operator!=(const sentinel& x, const iterator<Const>& y);[...] }; }
and from 23.7.5.4 [range.transform.sentinel], paragraphs 4-7:
Effects: Equivalent to:friend constexpr bool operator==(const iterator<Const>& x, const sentinel& y);
return x.current_ == y.end_
;
friend constexpr bool operator==(const sentinel& x, const iterator<Const>& y);Effects: Equivalent to:return y == x;
friend constexpr bool operator!=(const iterator<Const>& x, const sentinel& y);Effects: Equivalent to:return !(x == y);
friend constexpr bool operator!=(const sentinel& x, const iterator<Const>& y);Effects: Equivalent to:return !(y == x);
Remove obsolete equality operators from 23.7.6.3 [range.take.sentinel]:
namespace std::ranges { template<class V> template<bool Const> class take_view<V>::sentinel { private: using Base = conditional_t<Const, const V, V>; // exposition only using CI = counted_iterator<iterator_t<Base>>; // exposition only sentinel_t<Base> end_ = sentinel_t<Base>(); // exposition only public: [...]
friend constexpr bool operator==(const sentinel& x, const CI& y);friend constexpr bool operator==(const CI& y, const sentinel& x);friend constexpr bool operator!=(const sentinel& x, const CI& y);friend constexpr bool operator!=(const CI& y, const sentinel& x);}; }
and from 23.7.6.3 [range.take.sentinel] paragraphs 4-5:
Effects: Equivalent to:
friend constexpr bool operator==(const sentinel& x, const CI& y);friend constexpr bool operator==(const CI& y, const sentinel& x);return y.count() == 0 || y.base() == x.end_;
friend constexpr bool operator!=(const sentinel& x, const CI& y); friend constexpr bool operator!=(const CI& y, const sentinel& x);Effects: Equivalent to:return !(x == y);
Remove obsolete operator!=
from 23.7.7.3 [range.join.iterator]:
namespace std::ranges { template<class V> template<bool Const> struct join_view<V>::iterator { [...] friend constexpr bool operator==(const iterator& x, const iterator& y) requires ref_is_glvalue && EqualityComparable<iterator_t<Base>> && EqualityComparable<iterator_t<iter_reference_t<iterator_t<Base>>>>;
friend constexpr bool operator!=(const iterator& x, const iterator& y) requires ref_is_glvalue && EqualityComparable<iterator_t<Base>> && EqualityComparable<iterator_t<iter_reference_t<iterator_t<Base>>>>;[...] }; }
and from 23.7.7.3 [range.join.iterator] paragraph 17:
Effects: Equivalent to:friend constexpr bool operator==(const iterator& x, const iterator& y) requires ref_is_glvalue && EqualityComparable<iterator_t<Base>> && EqualityComparable<iterator_t<iter_reference_t<iterator_t<Base>>>>;
return x.outer_ == y.outer_ && x.inner_ == y.inner_;
friend constexpr bool operator!=(const iterator& x, const iterator& y) requires ref_is_glvalue && EqualityComparable<iterator_t<Base>> && EqualityComparable<iterator_t<iter_reference_t<iterator_t<Base>>>>;Effects: Equivalent to:return !(x == y);
Remove obsolete equality operators from 23.7.7.4 [range.join.sentinel]:
namespace std::ranges { template<class V> template<bool Const> struct join_view<V>::sentinel { [...] friend constexpr bool operator==(const iterator<Const>& x, const sentinel& y);
friend constexpr bool operator==(const sentinel& x, const iterator<Const>& y);friend constexpr bool operator!=(const iterator<Const>& x, const sentinel& y);friend constexpr bool operator!=(const sentinel& x, const iterator<Const>& y);}; }
and from 23.7.7.4 [range.join.sentinel] paragraphs 3-6:
Effects: Equivalent to:friend constexpr bool operator==(const iterator<Const>& x, const sentinel& y);
return x.outer_ == y.end_;
friend constexpr bool operator==(const sentinel& x, const iterator<Const>& y);Effects: Equivalent to:return y == x;
friend constexpr bool operator!=(const iterator<Const>& x, const sentinel& y);Effects: Equivalent to:return !(x == y);
friend constexpr bool operator!=(const sentinel& x, const iterator<Const>& y);Effects: Equivalent to:return !(y == x);
Remove obsolete equality operators from 23.7.8.3 [range.split.outer]:
namespace std::ranges { template<class V, class Pattern> template<bool Const> struct split_view<V, Pattern>::outer_iterator { [...] friend constexpr bool operator==(const outer_iterator& x, const outer_iterator& y) requires ForwardRange<Base>;
friend constexpr bool operator!=(const outer_iterator& x, const outer_iterator& y)requires ForwardRange<Base>;friend constexpr bool operator==(const outer_iterator& x, default_sentinel_t);friend constexpr bool operator==(default_sentinel_t, const outer_iterator& x);friend constexpr bool operator!=(const outer_iterator& x, default_sentinel_t y);friend constexpr bool operator!=(default_sentinel_t y, const outer_iterator& x);}; }
and from 23.7.8.3 [range.split.outer] paragraphs 7-10:
Effects: Equivalent to:friend constexpr bool operator==(const outer_iterator& x, const outer_iterator& y) requires ForwardRange<Base>;
return x.current_ == y.current_;
friend constexpr bool operator!=(const outer_iterator& x, const outer_iterator& y) requires ForwardRange<Base>;Effects: Equivalent to:return !(x == y);
Effects: Equivalent to:friend constexpr bool operator==(const outer_iterator& x, default_sentinel_t);
friend constexpr bool operator==(default_sentinel_t, const outer_iterator& x);return x.current_ == ranges::end(x.parent_->base_);
friend constexpr bool operator!=(const outer_iterator& x, default_sentinel_t y); friend constexpr bool operator!=(default_sentinel_t y, const outer_iterator& x);Effects: Equivalent to:return !(x == y);
Remove obsolete equality operators from 23.7.8.5 [range.split.inner]:
namespace std::ranges { template<class V, class Pattern> template<bool Const> struct split_view<V, Pattern>::inner_iterator { [...] friend constexpr bool operator==(const inner_iterator& x, const inner_iterator& y) requires ForwardRange<Base>;
friend constexpr bool operator!=(const inner_iterator& x, const inner_iterator& y)requires ForwardRange<Base>;friend constexpr bool operator==(const inner_iterator& x, default_sentinel_t);friend constexpr bool operator==(default_sentinel_t, const inner_iterator& x);friend constexpr bool operator!=(const inner_iterator& x, default_sentinel_t y);friend constexpr bool operator!=(default_sentinel_t y, const inner_iterator& x);[...] }; }
and from 23.7.8.5 [ranges.split.inner] paragraphs 4-7:
Effects: Equivalent to:friend constexpr bool operator==(const inner_iterator& x, const inner_iterator& y) requires ForwardRange<Base>;
return x.i_.current_ == y.i_.current_;
friend constexpr bool operator!=(const inner_iterator& x, const inner_iterator& y) requires ForwardRange<Base>;Effects: Equivalent to:return !(x == y);
Effects: Equivalent to:friend constexpr bool operator==(const inner_iterator& x, default_sentinel_t);
friend constexpr bool operator==(default_sentinel_t, const inner_iterator& x);auto cur = x.i_.current; auto end = ranges::end(x.i_.parent_->base_); if (cur == end) return true; auto [pcur, pend] = subrange{x.i_.parent_->pattern_}; if (pcur == pend) return x.incremented_; do { if (cur != pcur) return false; if (++pcur == pend) return true; } while (++cur != end); return false;
friend constexpr bool operator!=(const inner_iterator& x, default_sentinel_t y); friend constexpr bool operator!=(default_sentinel_t y, const inner_iterator& x);Effects: Equivalent to:return !(x == y);
Change 24.4 [algorithm.syn]:
namespace std { [...] // [alg.3way], three-way comparison algorithms
template<class T, class U>constexpr auto compare_3way(const T& a, const U& b);template<class InputIterator1, class InputIterator2, class Cmp> constexpr autolexicographical_compare_3way(InputIterator1 b1, InputIterator1 e1,lexicographical_compare_three_way(InputIterator1 b1, InputIterator1 e1, InputIterator2 b2, InputIterator2 e2, Cmp comp) -> common_comparison_category_t<decltype(comp(*b1, *b2)), strong_ordering>; template<class InputIterator1, class InputIterator2> constexpr autolexicographical_compare_3way(InputIterator1 b1, InputIterator1 e1,lexicographical_compare_three_way(InputIterator1 b1, InputIterator1 e1, InputIterator2 b2, InputIterator2 e2); [...] }
Change 24.7.11 [alg.3way]:
template<class T, class U> constexpr auto compare_3way(const T& a, const U& b);
Effects: Compares two values and produces a result of the strongest applicable comparison category type:
- Returns a <=> b if that expression is well-formed.
- Otherwise, if the expressions a == b and a < b are each well-formed and convertible to bool, returns strong_ordering::equal when a == b is true, otherwise returns strong_ordering::less when a < b is true, and otherwise returns strong_ordering::greater.
- Otherwise, if the expression a == b is well-formed and convertible to bool, returns strong_equality::equal when a == b is true, and otherwise returns strong_equality::nonequal.
- Otherwise, the function is defined as deleted.
Change 24.7.11 [alg.3way] paragraph 2:
template<class InputIterator1, class InputIterator2, class Cmp> constexpr auto
lexicographical_compare_3way(InputIterator1 b1, InputIterator1 e1,lexicographical_compare_three_way(InputIterator1 b1, InputIterator1 e1, InputIterator2 b2, InputIterator2 e2, Cmp comp);
Change 24.7.11 [alg.3way] paragraph 4:
Effects: Equivalent to:template<class InputIterator1, class InputIterator2> constexpr auto
lexicographical_compare_3way(InputIterator1 b1, InputIterator1 e1,lexicographical_compare_three_way(InputIterator1 b1, InputIterator1 e1, InputIterator2 b2, InputIterator2 e2);
return lexicographical_compare_3way(b1, e1, b2, e2,[](const auto& t, const auto& u) {return compare_3way(t, u);});return lexicographical_compare_three_way(b1, e1, b2, e2, compare_three_way());
No changes.
TBD
TBD
TBD
TBD
TBD
TBD