std::cmp_equal, cmp_not_equal, cmp_less, cmp_greater, cmp_less_equal, cmp_greater_equal

来自cppreference.com
< cpp‎ | utility
 
 
工具库
通用工具
日期和时间
函数对象
格式化库 (C++20)
(C++11)
关系运算符 (C++20 中弃用)
整数比较函数
cmp_equalcmp_lesscmp_less_than
(C++20)(C++20)(C++20)
cmp_not_equalcmp_greatercmp_greater_than
(C++20)(C++20)(C++20)
(C++20)
swap 与类型运算
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
常用词汇类型
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)
初等字符串转换
(C++17)
(C++17)
 
在标头 <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 起)

比较二个整数 tu 的值。不同于内建比较运算符,负有符号整数始终比较小于(且不等于)无符号整数:该比较相对于一般算术转换是安全的。

-1 > 0u; // true
std::cmp_greater(-1, 0u); // false

TU 不是有符号或无符号整数类型(包括标准整数类型与扩展整数类型),则为编译时错误。

参数

t - 左侧参数
u - 右侧参数

返回值

1)t 等于 u 则为 true
2)t 不等于 u 则为 true
3)t 小于 u 则为 true
4)t 大于 u 则为 true
5)t 小于或等于 u 则为 true
6)t 大于或等于 u 则为 true

可能的实现

template< class T, class U >
constexpr bool cmp_equal( T t, U u ) noexcept
{
    using UT = std::make_unsigned_t<T>;
    using UU = std::make_unsigned_t<U>;
    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 ? false : UT(t) == u;
    else
        return u < 0 ? false : t == UU(u);
}
 
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
{
    using UT = std::make_unsigned_t<T>;
    using UU = std::make_unsigned_t<U>;
    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 ? true : UT(t) < u;
    else
        return u < 0 ? false : t < UU(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_greater(t, u);
}
 
template< class T, class U >
constexpr bool cmp_greater_equal( T t, U u ) noexcept
{
    return !cmp_less(t, u);
}

注解

这些函数不能用于比较枚举(包括 std::byte )、 charchar8_tchar16_tchar32_twchar_tbool

功能特性测试
__cpp_lib_integer_comparison_functions

示例

如果编译时没有添加适当的警告抑制参数,下面的例子可能会产生不同符号性比较警告, 例如 -Wno-sign-compare (gcc/clang 使用 -Wall -Wextra, 另外参见 SO: disabling a specific warning)。

#include <utility>
 
// 取消下一行的注释将禁用 "有符号/无符号比较 "的警告:
// #pragma GCC diagnostic ignored "-Wsign-compare"
 
int main()
{
    static_assert( sizeof(int) == 4 ); // 先决条件
 
    // 意外
    static_assert( -1 > 1U ); //< 警告:无符号-有符号比较
    // 因为在经过隐式转换后,-1 的类型会变成右操作数的类型 (`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 ) );
}