std::add_sat

来自cppreference.com
< cpp‎ | numeric
在标头 <numeric> 定义
template< class T >
constexpr T add_sat( T x, T y ) noexcept;
(C++26 起)

计算饱和加法 x + y。该运算(与内置整数算术运算不同)的行为如同是一种具有 无穷 范围的数学运算。令 q 表示这种运算的结果。 返回:

  • q,如果 q 表示 T 类型的值。否则,
  • T 类型最接近 q 的最大值或者最小值。

该重载仅当 T整数类型时参与重载决议,包括:signed charshortintlonglong long 和扩展有符号整数类型,以及上述类型的无符号版本。特别是,T 不能是(可能 cv 限定的)boolcharwchar_tchar8_tchar16_tchar32_t,因为这些类型不用于算术运算。

参数

x, y - 整数值

返回值

饱和的 x + y

注解

与内置的整数算术运算符不同,并不对实参 xy 应用整数提升

如果传递了两个不同类型的实参,调用将无法编译,即其与模板实参推导相关的行为与 std::minstd::max 相同。

大多数现代硬件架构都能有效支持 SIMD 向量上的饱和运算,包括 x86 的 SSE2 和 ARM 的 NEON。

功能特性测试 标准 功能特性
__cpp_lib_saturation_arithmetic 202311L (C++26) 饱和算数

可能的实现

参考 libstdc++ (gcc)

示例

可以在 Compiler Explorer 预览

#include <climits>
#include <limits>
#include <numeric>
 
static_assert(CHAR_BIT == 8);
static_assert(UCHAR_MAX == 255);
 
int main()
{
    constexpr int a = std::add_sat(3, 4); // 未饱和, T = int
    static_assert(a == 7);
 
    constexpr unsigned char b = std::add_sat<unsigned char>(UCHAR_MAX, 4); // 饱和
    static_assert(b == UCHAR_MAX);
 
    constexpr unsigned char c = std::add_sat(UCHAR_MAX, 4); // 未饱和, T = int
        // add_sat(int, int) 返回 int tmp == 259,
        // 然后赋值时截断 259 % 256 == 3
    static_assert(c == 3);
 
//  unsigned char d = std::add_sat(252, c); // 错误:T 的推导不一致
 
    constexpr unsigned char e = std::add_sat<unsigned char>(251, a); // 饱和
    static_assert(e == UCHAR_MAX);
        // 251 类型为 T = unsigned char, `a` 被转换为 unsigned char 值;
        // 可能引发 `a` 的 int -> unsigned char 转换警告
 
    constexpr signed char f = std::add_sat<signed char>(-123, -3); // 未饱和
    static_assert(f == -126);
 
    constexpr signed char g = std::add_sat<signed char>(-123, -13); // 饱和
    static_assert(g == std::numeric_limits<signed char>::min()); // g == -128
}

参阅

(C++26)
两个整数的饱和减法运算
(函数模板)
(C++26)
两个整数的饱和乘法运算
(函数模板)
(C++26)
两个整数的饱和除法运算
(函数模板)
返回在另一个整数类型范围内的整数值
(函数模板)
(C++17)
在一对边界值间夹逼一个值
(函数模板)
(C++20)
检查整数值是否在给定整数类型的范围内
(函数模板)
[静态]
返回给定类型的最小有限值
(std::numeric_limits<T> 的公开静态成员函数)
[静态]
返回给定类型的最大有限值
(std::numeric_limits<T> 的公开静态成员函数)

外部链接

1.  A branch-free implementation of saturation arithmetic — Locklessinc.com, 2012