std::cmp_equal, cmp_not_equal, cmp_less, cmp_greater, cmp_less_equal, cmp_greater_equal
| Определено в заголовочном файле <utility>
|
||
template< class T, class U > constexpr bool cmp_equal( T t, U u ) noexcept; |
(1) | (начиная с C++20) |
template< class T, class U > constexpr bool cmp_not_equal( T t, U u ) noexcept; |
(2) | (начиная с C++20) |
template< class T, class U > constexpr bool cmp_less( T t, U u ) noexcept; |
(3) | (начиная с C++20) |
template< class T, class U > constexpr bool cmp_greater( T t, U u ) noexcept; |
(4) | (начиная с C++20) |
template< class T, class U > constexpr bool cmp_less_equal( T t, U u ) noexcept; |
(5) | (начиная с C++20) |
template< class T, class U > constexpr bool cmp_greater_equal( T t, U u ) noexcept; |
(6) | (начиная с C++20) |
Сравнивает значения двух целых чисел t и u. В отличие от встроенных операторов сравнения, отрицательные целые числа со знаком всегда при сравнении меньше (и не равны) целых чисел без знака: сравнение защищено от преобразования целых чисел с потерями.
-1 > 0u; // true
std::cmp_greater(-1, 0u); // false
Это ошибка времени компиляции, если T или U не являются целыми числами со знаком или без знака (включая стандартный целочисленный тип и расширенный целочисленный тип).
Параметры
| t | — | левый аргумент |
| u | — | правый аргумент |
Возвращаемое значение
true, если t равна u.true, если t не равна u.true, если t меньше чем u.true, если t больше чем u.true, если t меньше или равна u.true, если t больше или равна u.Возможная реализация
template<class T, class U>
constexpr bool cmp_equal(T t, U u) noexcept
{
if constexpr (std::is_signed_v<T> == std::is_signed_v<U>)
return t == u;
else if constexpr (std::is_signed_v<T>)
return t >= 0 && std::make_unsigned_t<T>(t) == u;
else
return u >= 0 && std::make_unsigned_t<U>(u) == t;
}
template<class T, class U>
constexpr bool cmp_not_equal(T t, U u) noexcept
{
return !cmp_equal(t, u);
}
template<class T, class U>
constexpr bool cmp_less(T t, U u) noexcept
{
if constexpr (std::is_signed_v<T> == std::is_signed_v<U>)
return t < u;
else if constexpr (std::is_signed_v<T>)
return t < 0 || std::make_unsigned_t<T>(t) < u;
else
return u >= 0 && t < std::make_unsigned_t<U>(u);
}
template<class T, class U>
constexpr bool cmp_greater(T t, U u) noexcept
{
return cmp_less(u, t);
}
template<class T, class U>
constexpr bool cmp_less_equal(T t, U u) noexcept
{
return !cmp_less(u, t);
}
template<class T, class U>
constexpr bool cmp_greater_equal(T t, U u) noexcept
{
return !cmp_less(t, u);
}
|
Примечание
Эти функции нельзя использовать для сравнения перечислений (включая std::byte), char, char8_t, char16_t, char32_t, wchar_t и bool.
| Макрос Тестирования функциональности | Значение | Стандарт | Функциональность |
|---|---|---|---|
__cpp_lib_integer_comparison_functions |
202002L |
(C++20) | no section name |
Пример
В приведённом ниже примере может появиться предупреждение different signedness comparison, если он скомпилирован без соответствующего флага подавления предупреждений, например, -Wno-sign-compare (gcc/clang с -Wall -Wextra, смотрите также SO: отключение конкретного предупреждения).
#include <utility>
// Раскомментирование следующей строки отключит предупреждения
// "знакового/беззнакового сравнения":
// #pragma GCC diagnostic ignored "-Wsign-compare"
int main()
{
static_assert( sizeof(int) == 4 ); // предварительное условие
// Совершенно неожиданно
static_assert( -1 > 1U ); //< предупреждение: сравнение знакового и беззнакового
// так как после неявного преобразования -1 в тип RHS (`unsigned int`)
// выражение эквивалентно:
static_assert( 0xFFFFFFFFU > 1U );
static_assert( 0xFFFFFFFFU == static_cast<unsigned>(-1) );
// Напротив, семейство cmp_* сравнивает целые числа, как и ожидалось,
// отрицательные целые числа со знаком всегда при сравнении меньше,
// чем целые числа без знака:
static_assert( std::cmp_less( -1, 1U ) );
static_assert( std::cmp_less_equal( -1, 1U ) );
static_assert( ! std::cmp_greater( -1, 1U ) );
static_assert( ! std::cmp_greater_equal( -1, 1U ) );
//< предупреждение: сравнение знакового и беззнакового
static_assert( -1 == 0xFFFFFFFFU );
static_assert( std::cmp_not_equal( -1, 0xFFFFFFFFU ) );
}
Смотрите также
функциональный объект, реализующий x == y (шаблон класса) | |
функциональный объект, реализующий x != y (шаблон класса) | |
функциональный объект, реализующий x < y (шаблон класса) | |
функциональный объект, реализующий x > y (шаблон класса) | |
функциональный объект, реализующий x <= y (шаблон класса) | |
функциональный объект, реализующий x >= y (шаблон класса) | |
(C++20) |
функциональный объект, реализующий x == y (класс) |
(C++20) |
функциональный объект, реализующий x != y (класс) |
(C++20) |
функциональный объект, реализующий x < y (класс) |
(C++20) |
функциональный объект, реализующий x > y (класс) |
(C++20) |
функциональный объект, реализующий x <= y (класс) |
(C++20) |
функциональный объект, реализующий x >= y (класс) |
(C++20) |
функциональный объект, реализующий x <=> y (класс) |
(C++20) |
проверяет, находится ли целочисленное значение в диапазоне заданного целочисленного типа (шаблон функции) |
| предоставляет интерфейс для запроса свойств всех основных числовых типов. (шаблон класса) |